351 lines
8.8 KiB
JavaScript
351 lines
8.8 KiB
JavaScript
const fetch = require('node-fetch');
|
|
const { Op, fn, col } = require('sequelize');
|
|
|
|
const BookReferenceController = require('./bookReference');
|
|
|
|
class ShelfController {
|
|
constructor (sequelizeModels, language) { // Language needs to be passed with every request involving books.
|
|
this.models = sequelizeModels;
|
|
this.lang = language;
|
|
}
|
|
|
|
static userOwnsShelf(user, shelf) {
|
|
return typeof user !== 'undefined' && user.id === shelf.userId;
|
|
}
|
|
|
|
static shelfCanBeModified(shelf) {
|
|
return shelf.isDeletable === true;
|
|
}
|
|
|
|
static newShelfNameIsValid (name, existingNames = []) {
|
|
if (name.length < 1) {
|
|
return {
|
|
error: true,
|
|
message: 'api.shelf.create.name_too_short',
|
|
};
|
|
}
|
|
|
|
if (existingNames.includes(name)) {
|
|
return {
|
|
error: true,
|
|
message: 'api.shelf.create.name_already_exists',
|
|
};
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static async CheckExternalDomainForShelf (domain, shelfId) {
|
|
const response = await fetch(`https://${domain}/api/shelf/get/${shelfId}/`).then(response => response.json());
|
|
// validate response somehow
|
|
return response;
|
|
}
|
|
|
|
async createDefaultShelves (user) {
|
|
try {
|
|
const defaultShelvesCreated = await this.models.Shelf.bulkCreate([
|
|
{
|
|
userId: user.id,
|
|
name: 'Reading',
|
|
isDeletable: false,
|
|
},
|
|
{
|
|
userId: user.id,
|
|
name: 'Want to Read',
|
|
isDeletable: false,
|
|
},
|
|
{
|
|
userId: user.id,
|
|
name: 'Finished',
|
|
isDeletable: false,
|
|
},
|
|
{
|
|
userId: user.id,
|
|
name: 'Did Not Finish',
|
|
isDeletable: false,
|
|
}
|
|
]);
|
|
|
|
if (defaultShelvesCreated.some(result => !result)) {
|
|
return {
|
|
error: true,
|
|
shelfResults: defaultShelvesCreated,
|
|
};
|
|
}
|
|
|
|
return defaultShelvesCreated;
|
|
} catch (error) {
|
|
return {
|
|
error,
|
|
}
|
|
}
|
|
}
|
|
|
|
async createShelf (user, name) {
|
|
try {
|
|
return await user.addShelf({
|
|
name,
|
|
});
|
|
} catch(error) {
|
|
return {
|
|
error,
|
|
}
|
|
}
|
|
}
|
|
|
|
async renameShelf (user, shelf, name) {
|
|
try {
|
|
return await shelf.update({ name });
|
|
} catch(error) {
|
|
return {
|
|
error,
|
|
}
|
|
}
|
|
}
|
|
|
|
async getLastUpdatedTimestamp (shelf) {
|
|
const lastEditedItem = await shelf.getShelfItems({
|
|
attributes: ['updatedAt'],
|
|
order: [
|
|
[
|
|
'updatedAt',
|
|
'DESC'
|
|
],
|
|
],
|
|
limit: 1,
|
|
});
|
|
|
|
if (lastEditedItem.length > 0 && (lastEditedItem[0].updatedAt > shelf.updatedAt)) {
|
|
return lastEditedItem.updatedAt;
|
|
}
|
|
return shelf.updatedAt;
|
|
}
|
|
|
|
async getShelfById(shelfId) {
|
|
if (isNaN(parseInt(shelfId))) {
|
|
return {
|
|
error: 'Shelf ID Provided is not a number.',
|
|
};
|
|
}
|
|
|
|
// REMINDER: Factor in privacy levels
|
|
|
|
const shelf = await this.models.Shelf.findByPk(shelfId, {
|
|
include: [
|
|
{
|
|
as: 'User',
|
|
model: this.models.User,
|
|
attributes: ['id', 'username', 'displayName'],
|
|
required: true,
|
|
},
|
|
{
|
|
as: 'ShelfItems',
|
|
model: this.models.ShelfItem,
|
|
attributes: [
|
|
'id',
|
|
'bookEdition',
|
|
'order'
|
|
],
|
|
required: false,
|
|
include: [
|
|
{
|
|
as: 'Statuses',
|
|
model: this.models.Status,
|
|
attributes: ['id', 'text', 'progress', 'createdAt', 'updatedAt'],
|
|
required: false,
|
|
},
|
|
{
|
|
as: 'BookReference',
|
|
model: this.models.BookReference,
|
|
attributes: ['id', 'name', 'description', 'sources', 'covers'],
|
|
required: true,
|
|
},
|
|
],
|
|
orderBy: [[col('ShelfItems.order'), 'ASC']],
|
|
},
|
|
],
|
|
attributes: [
|
|
[col('Shelf.id'), 'id'],
|
|
'name',
|
|
'userId',
|
|
'permissionLevel',
|
|
'isDeletable',
|
|
'createdAt',
|
|
'updatedAt',
|
|
[fn('COUNT', col('ShelfItems.id')), 'numShelfItems'],
|
|
],
|
|
group: [
|
|
col('Shelf.id'),
|
|
col('User.id'),
|
|
col('ShelfItems.id'),
|
|
col('ShelfItems->Statuses.id'),
|
|
col('ShelfItems->BookReference.id'),
|
|
],
|
|
});
|
|
if (shelf === null) {
|
|
return {
|
|
error: `Shelf with ID ${shelfId} not found.`,
|
|
};
|
|
}
|
|
|
|
shelf.updatedAt = this.getLastUpdatedTimestamp(shelf);
|
|
shelf.isPublic = shelf.permissionLevel === 0;
|
|
|
|
return shelf;
|
|
}
|
|
|
|
getFakeShelf () {
|
|
const faker = require('faker');
|
|
const isOwnShelf = faker.random.boolean();
|
|
const fakeName = faker.random.boolean()
|
|
? faker.fake('{{name.firstName}} {{name.lastName}}')
|
|
: faker.fake('{{hacker.adjective}}{{hacker.noun}}');
|
|
|
|
const shelf = {
|
|
id: faker.random.number(),
|
|
userId: faker.random.number(),
|
|
user: isOwnShelf ? null : {
|
|
name: fakeName,
|
|
handle: faker.fake('@{{internet.userName}}@{{internet.domainName}}'),
|
|
},
|
|
name: faker.fake('{{hacker.ingverb}} {{hacker.noun}}'),
|
|
isPublic: Math.random() < 0.9,
|
|
updatedAt: faker.date.past(),
|
|
shelfItems: [],
|
|
};
|
|
|
|
const numberOfShelfItems = faker.random.number(50);
|
|
for (let i = 0; i < numberOfShelfItems; i++) {
|
|
const source = {
|
|
source: faker.internet.domainWord(),
|
|
sourceId: faker.random.uuid(),
|
|
};
|
|
const shelfItem = {
|
|
name: faker.lorem.words().split(' ').map(word => (word[0].toUpperCase() + word.substr(1))).join(' '),
|
|
author: faker.fake('a work by {{name.firstName}} {{name.lastName}}'),
|
|
source,
|
|
coverURL: faker.image.imageUrl(),
|
|
coverEdition: `img_${source.sourceId}`,
|
|
rating: faker.random.number(5),
|
|
review: faker.random.boolean()
|
|
? null
|
|
: faker.lorem.paragraph(),
|
|
}
|
|
|
|
shelf.shelfItems.push(shelfItem);
|
|
}
|
|
|
|
if (isOwnShelf) {
|
|
shelf.createdAt = faker.date.past(undefined, shelf.updatedAt);
|
|
shelf.isDeletable = true;
|
|
}
|
|
|
|
return shelf;
|
|
}
|
|
|
|
async userCanViewShelf (user, shelf) {
|
|
// This needs work when permissions are added.
|
|
const userOwnsShelf = ShelfController.userOwnsShelf(user, shelf);
|
|
console.log('owned?', userOwnsShelf);
|
|
console.log('isPublic?', shelf.isPublic);
|
|
return userOwnsShelf || shelf.isPublic;
|
|
}
|
|
|
|
async scrubShelfData (shelf, currentUser) {
|
|
const userOwnsShelf = currentUser.id === shelf.userId;
|
|
const shelfUser = userOwnsShelf ? null : shelf.User;
|
|
let user = {};
|
|
if (shelfUser !== null) {
|
|
user.name = shelfUser.displayName;
|
|
user.handle = shelfUser.username;
|
|
} else {
|
|
user = null;
|
|
}
|
|
|
|
// Untested
|
|
const shelfItems = await Promise.all(shelf.ShelfItems.map(async (shelfItem) => {
|
|
const bookReference = BookReferenceController.formatReferenceSources(shelfItem.BookReference);
|
|
const reviews = await bookReference.getInteractions({
|
|
where: {
|
|
userId: shelf.userId,
|
|
}, // Return all reviews for any bookEdition so they can be filtered on frontend.
|
|
attributes: ['text', 'rating', 'bookEdition'],
|
|
});
|
|
return {
|
|
title: bookReference.name,
|
|
author: bookReference.description,
|
|
sources: bookReference.sources,
|
|
covers: bookReference.covers,
|
|
updates: shelfItem.Statuses,
|
|
reviews,
|
|
};
|
|
}));
|
|
|
|
const shelfData = {
|
|
name: shelf.name,
|
|
user,
|
|
shelfItems,
|
|
};
|
|
|
|
return shelfData;
|
|
}
|
|
|
|
async addShelfItem(shelf, bookReferenceId, source = null) {
|
|
const bookReferenceController = new BookReferenceController(this.models, this.lang);
|
|
|
|
let bookId = bookReferenceId;
|
|
if (source !== null) {
|
|
const bookReference = await bookReferenceController.createOrUpdateReference(source, bookId);
|
|
bookId = bookReference.id;
|
|
}
|
|
|
|
if (shelf.ShelfItems.some(shelfItem => shelfItem.bookId === bookId)) {
|
|
return {
|
|
error: 'api.shelf.addItem.already_on_shelf', // This may need to change to account for editions later.
|
|
}
|
|
}
|
|
|
|
const shelfItem = await shelf.createShelfItem({ bookId }).catch(err => err);
|
|
|
|
if (!shelfItem) {
|
|
return {
|
|
error: shelfItem,
|
|
};
|
|
}
|
|
|
|
return shelfItem;
|
|
}
|
|
|
|
async moveShelfItem(shelfItem, toShelf) {
|
|
const success = await toShelf.addShelfItem(shelfItem);
|
|
|
|
if (!success) {
|
|
return {
|
|
error: shelfItem,
|
|
};
|
|
}
|
|
|
|
return success;
|
|
}
|
|
|
|
async deleteShelfItem(shelfItem) {
|
|
// Only fully remove if no statuses are associated
|
|
const statuses = await shelfItem.getStatuses();
|
|
const options = {};
|
|
if (statuses.length < 1) {
|
|
options.force = true;
|
|
}
|
|
|
|
const success = await shelfItem.destroy(options);
|
|
|
|
if (!success) {
|
|
return {
|
|
error: shelfItem,
|
|
};
|
|
}
|
|
|
|
return success;
|
|
}
|
|
}
|
|
|
|
module.exports = ShelfController; |