2023-09-17 22:29:39 +02:00
|
|
|
const { DataTypes, Model, where, fn, col } = require('sequelize')
|
2024-09-01 22:08:56 +02:00
|
|
|
const parseNameString = require('../utils/parsers/parseNameString')
|
2023-07-05 01:14:44 +02:00
|
|
|
|
2023-08-15 01:22:38 +02:00
|
|
|
class Author extends Model {
|
|
|
|
constructor(values, options) {
|
|
|
|
super(values, options)
|
2023-07-05 01:14:44 +02:00
|
|
|
|
2023-08-15 01:22:38 +02:00
|
|
|
/** @type {UUIDV4} */
|
|
|
|
this.id
|
|
|
|
/** @type {string} */
|
|
|
|
this.name
|
|
|
|
/** @type {string} */
|
|
|
|
this.lastFirst
|
|
|
|
/** @type {string} */
|
|
|
|
this.asin
|
|
|
|
/** @type {string} */
|
|
|
|
this.description
|
|
|
|
/** @type {string} */
|
|
|
|
this.imagePath
|
|
|
|
/** @type {UUIDV4} */
|
|
|
|
this.libraryId
|
|
|
|
/** @type {Date} */
|
|
|
|
this.updatedAt
|
|
|
|
/** @type {Date} */
|
|
|
|
this.createdAt
|
|
|
|
}
|
2023-07-05 01:14:44 +02:00
|
|
|
|
2024-09-01 22:08:56 +02:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {string} name
|
|
|
|
* @returns {string}
|
|
|
|
*/
|
|
|
|
static getLastFirst(name) {
|
|
|
|
if (!name) return null
|
|
|
|
return parseNameString.nameToLastFirst(name)
|
|
|
|
}
|
|
|
|
|
2023-08-18 21:40:36 +02:00
|
|
|
/**
|
|
|
|
* Check if author exists
|
2024-05-29 00:24:02 +02:00
|
|
|
* @param {string} authorId
|
2023-08-18 21:40:36 +02:00
|
|
|
* @returns {Promise<boolean>}
|
|
|
|
*/
|
|
|
|
static async checkExistsById(authorId) {
|
|
|
|
return (await this.count({ where: { id: authorId } })) > 0
|
|
|
|
}
|
|
|
|
|
2023-09-03 17:49:02 +02:00
|
|
|
/**
|
2024-08-31 20:27:48 +02:00
|
|
|
* Get author by name and libraryId. name case insensitive
|
2023-09-03 17:49:02 +02:00
|
|
|
* TODO: Look for authors ignoring punctuation
|
2024-05-29 00:24:02 +02:00
|
|
|
*
|
|
|
|
* @param {string} authorName
|
|
|
|
* @param {string} libraryId
|
2024-08-31 20:27:48 +02:00
|
|
|
* @returns {Promise<Author>}
|
2023-09-03 17:49:02 +02:00
|
|
|
*/
|
2024-08-31 20:27:48 +02:00
|
|
|
static async getByNameAndLibrary(authorName, libraryId) {
|
2024-09-13 19:09:32 +02:00
|
|
|
return this.findOne({
|
|
|
|
where: [
|
2024-10-08 21:52:26 +02:00
|
|
|
where(fn('lower', col('name')), authorName.toLowerCase()),
|
2024-09-13 19:09:32 +02:00
|
|
|
{
|
|
|
|
libraryId
|
2024-08-31 20:27:48 +02:00
|
|
|
}
|
2024-09-13 19:09:32 +02:00
|
|
|
]
|
|
|
|
})
|
2023-09-03 17:49:02 +02:00
|
|
|
}
|
|
|
|
|
2024-06-27 23:32:38 +02:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {string} authorId
|
|
|
|
* @returns {Promise<import('./LibraryItem')[]>}
|
|
|
|
*/
|
|
|
|
static async getAllLibraryItemsForAuthor(authorId) {
|
|
|
|
const author = await this.findByPk(authorId, {
|
|
|
|
include: [
|
|
|
|
{
|
|
|
|
model: this.sequelize.models.book,
|
|
|
|
include: [
|
|
|
|
{
|
|
|
|
model: this.sequelize.models.libraryItem
|
|
|
|
},
|
|
|
|
{
|
|
|
|
model: this.sequelize.models.author,
|
|
|
|
through: {
|
|
|
|
attributes: []
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{
|
|
|
|
model: this.sequelize.models.series,
|
|
|
|
through: {
|
|
|
|
attributes: ['sequence']
|
|
|
|
}
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
})
|
|
|
|
|
|
|
|
const libraryItems = []
|
|
|
|
if (author.books) {
|
|
|
|
for (const book of author.books) {
|
|
|
|
const libraryItem = book.libraryItem
|
|
|
|
libraryItem.media = book
|
|
|
|
delete book.libraryItem
|
|
|
|
libraryItems.push(libraryItem)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return libraryItems
|
|
|
|
}
|
|
|
|
|
2025-01-05 19:05:01 +01:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {string} name
|
|
|
|
* @param {string} libraryId
|
|
|
|
* @returns {Promise<Author>}
|
|
|
|
*/
|
|
|
|
static async findOrCreateByNameAndLibrary(name, libraryId) {
|
|
|
|
const author = await this.getByNameAndLibrary(name, libraryId)
|
|
|
|
if (author) return author
|
|
|
|
return this.create({
|
|
|
|
name,
|
|
|
|
lastFirst: this.getLastFirst(name),
|
|
|
|
libraryId
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-08-15 01:22:38 +02:00
|
|
|
/**
|
|
|
|
* Initialize model
|
2024-05-29 00:24:02 +02:00
|
|
|
* @param {import('../Database').sequelize} sequelize
|
2023-08-15 01:22:38 +02:00
|
|
|
*/
|
|
|
|
static init(sequelize) {
|
2024-05-29 00:24:02 +02:00
|
|
|
super.init(
|
|
|
|
{
|
|
|
|
id: {
|
|
|
|
type: DataTypes.UUID,
|
|
|
|
defaultValue: DataTypes.UUIDV4,
|
|
|
|
primaryKey: true
|
2023-08-15 01:22:38 +02:00
|
|
|
},
|
2024-05-29 00:24:02 +02:00
|
|
|
name: DataTypes.STRING,
|
|
|
|
lastFirst: DataTypes.STRING,
|
|
|
|
asin: DataTypes.STRING,
|
|
|
|
description: DataTypes.TEXT,
|
|
|
|
imagePath: DataTypes.STRING
|
|
|
|
},
|
|
|
|
{
|
|
|
|
sequelize,
|
|
|
|
modelName: 'author',
|
|
|
|
indexes: [
|
|
|
|
{
|
|
|
|
fields: [
|
|
|
|
{
|
|
|
|
name: 'name',
|
|
|
|
collate: 'NOCASE'
|
|
|
|
}
|
|
|
|
]
|
|
|
|
},
|
|
|
|
// {
|
|
|
|
// fields: [{
|
|
|
|
// name: 'lastFirst',
|
|
|
|
// collate: 'NOCASE'
|
|
|
|
// }]
|
|
|
|
// },
|
|
|
|
{
|
|
|
|
fields: ['libraryId']
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
)
|
2023-07-08 16:57:32 +02:00
|
|
|
|
2023-08-15 01:22:38 +02:00
|
|
|
const { library } = sequelize.models
|
|
|
|
library.hasMany(Author, {
|
|
|
|
onDelete: 'CASCADE'
|
|
|
|
})
|
|
|
|
Author.belongsTo(library)
|
|
|
|
}
|
2024-08-31 20:27:48 +02:00
|
|
|
|
|
|
|
toOldJSON() {
|
|
|
|
return {
|
|
|
|
id: this.id,
|
|
|
|
asin: this.asin,
|
|
|
|
name: this.name,
|
|
|
|
description: this.description,
|
|
|
|
imagePath: this.imagePath,
|
|
|
|
libraryId: this.libraryId,
|
|
|
|
addedAt: this.createdAt.valueOf(),
|
|
|
|
updatedAt: this.updatedAt.valueOf()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param {number} numBooks
|
|
|
|
* @returns
|
|
|
|
*/
|
|
|
|
toOldJSONExpanded(numBooks = 0) {
|
|
|
|
const oldJson = this.toOldJSON()
|
|
|
|
oldJson.numBooks = numBooks
|
|
|
|
return oldJson
|
|
|
|
}
|
|
|
|
|
|
|
|
toJSONMinimal() {
|
|
|
|
return {
|
|
|
|
id: this.id,
|
|
|
|
name: this.name
|
|
|
|
}
|
|
|
|
}
|
2023-08-15 01:22:38 +02:00
|
|
|
}
|
|
|
|
module.exports = Author
|