2024-12-01 19:44:21 +01:00
|
|
|
const { expect } = require('chai')
|
|
|
|
const { Sequelize } = require('sequelize')
|
|
|
|
const sinon = require('sinon')
|
|
|
|
|
|
|
|
const Database = require('../../../server/Database')
|
|
|
|
const ApiRouter = require('../../../server/routers/ApiRouter')
|
|
|
|
const LibraryItemController = require('../../../server/controllers/LibraryItemController')
|
|
|
|
const ApiCacheManager = require('../../../server/managers/ApiCacheManager')
|
|
|
|
const Logger = require('../../../server/Logger')
|
|
|
|
|
|
|
|
describe('LibraryItemController', () => {
|
|
|
|
/** @type {ApiRouter} */
|
|
|
|
let apiRouter
|
|
|
|
|
|
|
|
beforeEach(async () => {
|
|
|
|
global.ServerSettings = {}
|
|
|
|
Database.sequelize = new Sequelize({ dialect: 'sqlite', storage: ':memory:', logging: false })
|
|
|
|
Database.sequelize.uppercaseFirst = (str) => (str ? `${str[0].toUpperCase()}${str.substr(1)}` : '')
|
|
|
|
await Database.buildModels()
|
|
|
|
|
|
|
|
apiRouter = new ApiRouter({
|
2024-12-15 19:38:50 +01:00
|
|
|
apiCacheManager: new ApiCacheManager()
|
2024-12-01 19:44:21 +01:00
|
|
|
})
|
|
|
|
|
|
|
|
sinon.stub(Logger, 'info')
|
|
|
|
})
|
|
|
|
|
|
|
|
afterEach(async () => {
|
|
|
|
sinon.restore()
|
|
|
|
|
|
|
|
// Clear all tables
|
|
|
|
await Database.sequelize.sync({ force: true })
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('checkRemoveAuthorsAndSeries', () => {
|
|
|
|
let libraryItem1Id
|
|
|
|
let libraryItem2Id
|
|
|
|
let author1Id
|
|
|
|
let author2Id
|
|
|
|
let author3Id
|
|
|
|
let series1Id
|
|
|
|
let series2Id
|
|
|
|
|
|
|
|
beforeEach(async () => {
|
|
|
|
const newLibrary = await Database.libraryModel.create({ name: 'Test Library', mediaType: 'book' })
|
|
|
|
const newLibraryFolder = await Database.libraryFolderModel.create({ path: '/test', libraryId: newLibrary.id })
|
|
|
|
|
|
|
|
const newBook = await Database.bookModel.create({ title: 'Test Book', audioFiles: [], tags: [], narrators: [], genres: [], chapters: [] })
|
|
|
|
const newLibraryItem = await Database.libraryItemModel.create({ libraryFiles: [], mediaId: newBook.id, mediaType: 'book', libraryId: newLibrary.id, libraryFolderId: newLibraryFolder.id })
|
|
|
|
libraryItem1Id = newLibraryItem.id
|
|
|
|
|
|
|
|
const newBook2 = await Database.bookModel.create({ title: 'Test Book 2', audioFiles: [], tags: [], narrators: [], genres: [], chapters: [] })
|
|
|
|
const newLibraryItem2 = await Database.libraryItemModel.create({ libraryFiles: [], mediaId: newBook2.id, mediaType: 'book', libraryId: newLibrary.id, libraryFolderId: newLibraryFolder.id })
|
|
|
|
libraryItem2Id = newLibraryItem2.id
|
|
|
|
|
|
|
|
const newAuthor = await Database.authorModel.create({ name: 'Test Author', libraryId: newLibrary.id })
|
|
|
|
author1Id = newAuthor.id
|
|
|
|
const newAuthor2 = await Database.authorModel.create({ name: 'Test Author 2', libraryId: newLibrary.id })
|
|
|
|
author2Id = newAuthor2.id
|
|
|
|
const newAuthor3 = await Database.authorModel.create({ name: 'Test Author 3', imagePath: '/fake/path/author.png', libraryId: newLibrary.id })
|
|
|
|
author3Id = newAuthor3.id
|
|
|
|
|
|
|
|
// Book 1 has Author 1, Author 2 and Author 3
|
|
|
|
await Database.bookAuthorModel.create({ bookId: newBook.id, authorId: newAuthor.id })
|
|
|
|
await Database.bookAuthorModel.create({ bookId: newBook.id, authorId: newAuthor2.id })
|
|
|
|
await Database.bookAuthorModel.create({ bookId: newBook.id, authorId: newAuthor3.id })
|
|
|
|
|
|
|
|
// Book 2 has Author 2
|
|
|
|
await Database.bookAuthorModel.create({ bookId: newBook2.id, authorId: newAuthor2.id })
|
|
|
|
|
|
|
|
const newSeries = await Database.seriesModel.create({ name: 'Test Series', libraryId: newLibrary.id })
|
|
|
|
series1Id = newSeries.id
|
|
|
|
const newSeries2 = await Database.seriesModel.create({ name: 'Test Series 2', libraryId: newLibrary.id })
|
|
|
|
series2Id = newSeries2.id
|
|
|
|
|
|
|
|
// Book 1 is in Series 1 and Series 2
|
|
|
|
await Database.bookSeriesModel.create({ bookId: newBook.id, seriesId: newSeries.id })
|
|
|
|
await Database.bookSeriesModel.create({ bookId: newBook.id, seriesId: newSeries2.id })
|
|
|
|
|
|
|
|
// Book 2 is in Series 2
|
|
|
|
await Database.bookSeriesModel.create({ bookId: newBook2.id, seriesId: newSeries2.id })
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should remove authors and series with no books on library item delete', async () => {
|
|
|
|
const oldLibraryItem = await Database.libraryItemModel.getOldById(libraryItem1Id)
|
|
|
|
|
|
|
|
const fakeReq = {
|
|
|
|
query: {},
|
|
|
|
libraryItem: oldLibraryItem
|
|
|
|
}
|
|
|
|
const fakeRes = {
|
|
|
|
sendStatus: sinon.spy()
|
|
|
|
}
|
|
|
|
await LibraryItemController.delete.bind(apiRouter)(fakeReq, fakeRes)
|
|
|
|
|
|
|
|
expect(fakeRes.sendStatus.calledWith(200)).to.be.true
|
|
|
|
|
|
|
|
// Author 1 should be removed because it has no books
|
|
|
|
const author1Exists = await Database.authorModel.checkExistsById(author1Id)
|
|
|
|
expect(author1Exists).to.be.false
|
|
|
|
|
|
|
|
// Author 2 should not be removed because it still has Book 2
|
|
|
|
const author2Exists = await Database.authorModel.checkExistsById(author2Id)
|
|
|
|
expect(author2Exists).to.be.true
|
|
|
|
|
|
|
|
// Author 3 should not be removed because it has an image
|
|
|
|
const author3Exists = await Database.authorModel.checkExistsById(author3Id)
|
|
|
|
expect(author3Exists).to.be.true
|
|
|
|
|
|
|
|
// Series 1 should be removed because it has no books
|
|
|
|
const series1Exists = await Database.seriesModel.checkExistsById(series1Id)
|
|
|
|
expect(series1Exists).to.be.false
|
|
|
|
|
|
|
|
// Series 2 should not be removed because it still has Book 2
|
|
|
|
const series2Exists = await Database.seriesModel.checkExistsById(series2Id)
|
|
|
|
expect(series2Exists).to.be.true
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should remove authors and series with no books on library item batch delete', async () => {
|
|
|
|
// Batch delete library item 1
|
|
|
|
const fakeReq = {
|
|
|
|
query: {},
|
|
|
|
user: {
|
|
|
|
canDelete: true
|
|
|
|
},
|
|
|
|
body: {
|
|
|
|
libraryItemIds: [libraryItem1Id]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const fakeRes = {
|
|
|
|
sendStatus: sinon.spy()
|
|
|
|
}
|
|
|
|
await LibraryItemController.batchDelete.bind(apiRouter)(fakeReq, fakeRes)
|
|
|
|
|
|
|
|
expect(fakeRes.sendStatus.calledWith(200)).to.be.true
|
|
|
|
|
|
|
|
// Author 1 should be removed because it has no books
|
|
|
|
const author1Exists = await Database.authorModel.checkExistsById(author1Id)
|
|
|
|
expect(author1Exists).to.be.false
|
|
|
|
|
|
|
|
// Author 2 should not be removed because it still has Book 2
|
|
|
|
const author2Exists = await Database.authorModel.checkExistsById(author2Id)
|
|
|
|
expect(author2Exists).to.be.true
|
|
|
|
|
|
|
|
// Author 3 should not be removed because it has an image
|
|
|
|
const author3Exists = await Database.authorModel.checkExistsById(author3Id)
|
|
|
|
expect(author3Exists).to.be.true
|
|
|
|
|
|
|
|
// Series 1 should be removed because it has no books
|
|
|
|
const series1Exists = await Database.seriesModel.checkExistsById(series1Id)
|
|
|
|
expect(series1Exists).to.be.false
|
|
|
|
|
|
|
|
// Series 2 should not be removed because it still has Book 2
|
|
|
|
const series2Exists = await Database.seriesModel.checkExistsById(series2Id)
|
|
|
|
expect(series2Exists).to.be.true
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should remove authors and series with no books on library item update media', async () => {
|
|
|
|
const oldLibraryItem = await Database.libraryItemModel.getOldById(libraryItem1Id)
|
|
|
|
|
|
|
|
// Update library item 1 remove all authors and series
|
|
|
|
const fakeReq = {
|
|
|
|
query: {},
|
|
|
|
body: {
|
|
|
|
metadata: {
|
|
|
|
authors: [],
|
|
|
|
series: []
|
|
|
|
}
|
|
|
|
},
|
|
|
|
libraryItem: oldLibraryItem
|
|
|
|
}
|
|
|
|
const fakeRes = {
|
|
|
|
json: sinon.spy()
|
|
|
|
}
|
|
|
|
await LibraryItemController.updateMedia.bind(apiRouter)(fakeReq, fakeRes)
|
|
|
|
|
|
|
|
expect(fakeRes.json.calledOnce).to.be.true
|
|
|
|
|
|
|
|
// Author 1 should be removed because it has no books
|
|
|
|
const author1Exists = await Database.authorModel.checkExistsById(author1Id)
|
|
|
|
expect(author1Exists).to.be.false
|
|
|
|
|
|
|
|
// Author 2 should not be removed because it still has Book 2
|
|
|
|
const author2Exists = await Database.authorModel.checkExistsById(author2Id)
|
|
|
|
expect(author2Exists).to.be.true
|
|
|
|
|
|
|
|
// Author 3 should not be removed because it has an image
|
|
|
|
const author3Exists = await Database.authorModel.checkExistsById(author3Id)
|
|
|
|
expect(author3Exists).to.be.true
|
|
|
|
|
|
|
|
// Series 1 should be removed because it has no books
|
|
|
|
const series1Exists = await Database.seriesModel.checkExistsById(series1Id)
|
|
|
|
expect(series1Exists).to.be.false
|
|
|
|
|
|
|
|
// Series 2 should not be removed because it still has Book 2
|
|
|
|
const series2Exists = await Database.seriesModel.checkExistsById(series2Id)
|
|
|
|
expect(series2Exists).to.be.true
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|