generated from nhcarrigan/template
208c11d153
### Explanation _No response_ ### Issue _No response_ ### Attestations - [ ] I have read and agree to the [Code of Conduct](https://docs.nhcarrigan.com/community/coc/) - [ ] I have read and agree to the [Community Guidelines](https://docs.nhcarrigan.com/community/guide/). - [ ] My contribution complies with the [Contributor Covenant](https://docs.nhcarrigan.com/dev/covenant/). ### Dependencies - [ ] I have pinned the dependencies to a specific patch version. ### Style - [ ] I have run the linter and resolved any errors. - [ ] My pull request uses an appropriate title, matching the conventional commit standards. - [ ] My scope of feat/fix/chore/etc. correctly matches the nature of changes in my pull request. ### Tests - [ ] My contribution adds new code, and I have added tests to cover it. - [ ] My contribution modifies existing code, and I have updated the tests to reflect these changes. - [ ] All new and existing tests pass locally with my changes. - [ ] Code coverage remains at or above the configured threshold. ### Documentation _No response_ ### Versioning _No response_ Reviewed-on: #64
189 lines
5.5 KiB
TypeScript
189 lines
5.5 KiB
TypeScript
/**
|
|
* @copyright 2026 NHCarrigan
|
|
* @license Naomi's Public License
|
|
* @author Naomi Carrigan
|
|
*/
|
|
|
|
import { Manga, MangaStatus, CreateMangaDto, UpdateMangaDto } from "@library/shared-types";
|
|
import { prisma } from "../lib/prisma";
|
|
import {
|
|
validateUrl,
|
|
validateRating,
|
|
validateStringLength,
|
|
validateDataUrl,
|
|
MAX_LENGTHS,
|
|
} from "../utils/validation";
|
|
|
|
export class MangaService {
|
|
private prisma = prisma;
|
|
|
|
constructor() {}
|
|
|
|
/**
|
|
* Validate manga data for security.
|
|
*/
|
|
private validateMangaData(data: CreateMangaDto | UpdateMangaDto): void {
|
|
// Validate string lengths
|
|
if (!validateStringLength(data.title, MAX_LENGTHS.TITLE)) {
|
|
throw new Error(`Title must be ${MAX_LENGTHS.TITLE} characters or less.`);
|
|
}
|
|
if (!validateStringLength(data.author, MAX_LENGTHS.AUTHOR)) {
|
|
throw new Error(`Author must be ${MAX_LENGTHS.AUTHOR} characters or less.`);
|
|
}
|
|
if (!validateStringLength(data.notes, MAX_LENGTHS.NOTES)) {
|
|
throw new Error(`Notes must be ${MAX_LENGTHS.NOTES} characters or less.`);
|
|
}
|
|
if (!validateStringLength(data.coverImage, MAX_LENGTHS.URL)) {
|
|
throw new Error(`Cover image URL must be ${MAX_LENGTHS.URL} characters or less.`);
|
|
}
|
|
|
|
// Validate rating
|
|
if (!validateRating(data.rating)) {
|
|
throw new Error("Rating must be an integer between 0 and 10.");
|
|
}
|
|
|
|
// Validate cover image URL
|
|
if (data.coverImage) {
|
|
if (data.coverImage.startsWith("data:")) {
|
|
const sizeInBytes = data.coverImage.length * 0.75;
|
|
if (sizeInBytes > MAX_LENGTHS.DATA_URL) {
|
|
throw new Error("Cover image must be under 5MB.");
|
|
}
|
|
if (!validateDataUrl(data.coverImage)) {
|
|
throw new Error("Invalid image data URL.");
|
|
}
|
|
} else {
|
|
if (!validateStringLength(data.coverImage, MAX_LENGTHS.URL)) {
|
|
throw new Error(`Cover image URL must be ${MAX_LENGTHS.URL} characters or less.`);
|
|
}
|
|
if (!validateUrl(data.coverImage)) {
|
|
throw new Error("Invalid cover image URL. Only http and https URLs are allowed.");
|
|
}
|
|
}
|
|
}
|
|
|
|
// Validate tags
|
|
if (data.tags) {
|
|
for (const tag of data.tags) {
|
|
if (!validateStringLength(tag, MAX_LENGTHS.TAGS)) {
|
|
throw new Error(`Each tag must be ${MAX_LENGTHS.TAGS} characters or less.`);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Validate link URLs
|
|
if (data.links) {
|
|
for (const link of data.links) {
|
|
if (!validateUrl(link.url)) {
|
|
throw new Error(`Invalid link URL: ${link.title}. Only http and https URLs are allowed.`);
|
|
}
|
|
if (!validateStringLength(link.title, MAX_LENGTHS.TITLE)) {
|
|
throw new Error(`Link title must be ${MAX_LENGTHS.TITLE} characters or less.`);
|
|
}
|
|
if (!validateStringLength(link.url, MAX_LENGTHS.URL)) {
|
|
throw new Error(`Link URL must be ${MAX_LENGTHS.URL} characters or less.`);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
async getAllManga(): Promise<Manga[]> {
|
|
const manga = await this.prisma.manga.findMany({
|
|
orderBy: { updatedAt: "desc" },
|
|
});
|
|
|
|
return manga.map((m) => ({
|
|
...m,
|
|
status: m.status as unknown as MangaStatus,
|
|
dateAdded: m.dateAdded,
|
|
dateStarted: m.dateStarted || undefined,
|
|
dateCompleted: m.dateCompleted || undefined,
|
|
dateFinished: m.dateFinished || undefined,
|
|
tags: m.tags ?? [],
|
|
links: m.links ?? [],
|
|
createdAt: m.createdAt,
|
|
updatedAt: m.updatedAt,
|
|
}));
|
|
}
|
|
|
|
async getMangaById(id: string): Promise<Manga | null> {
|
|
const manga = await this.prisma.manga.findUnique({
|
|
where: { id },
|
|
});
|
|
|
|
if (!manga) return null;
|
|
|
|
return {
|
|
...manga,
|
|
status: manga.status as unknown as MangaStatus,
|
|
dateAdded: manga.dateAdded,
|
|
dateStarted: manga.dateStarted || undefined,
|
|
dateCompleted: manga.dateCompleted || undefined,
|
|
dateFinished: manga.dateFinished || undefined,
|
|
tags: manga.tags ?? [],
|
|
links: manga.links ?? [],
|
|
createdAt: manga.createdAt,
|
|
updatedAt: manga.updatedAt,
|
|
};
|
|
}
|
|
|
|
async createManga(data: CreateMangaDto): Promise<Manga> {
|
|
// Validate input
|
|
this.validateMangaData(data);
|
|
|
|
const manga = await this.prisma.manga.create({
|
|
data: {
|
|
...data,
|
|
status: data.status.toUpperCase() as any,
|
|
},
|
|
});
|
|
|
|
return {
|
|
...manga,
|
|
status: manga.status as unknown as MangaStatus,
|
|
dateAdded: manga.dateAdded,
|
|
dateStarted: manga.dateStarted || undefined,
|
|
dateCompleted: manga.dateCompleted || undefined,
|
|
dateFinished: manga.dateFinished || undefined,
|
|
tags: manga.tags ?? [],
|
|
links: manga.links ?? [],
|
|
createdAt: manga.createdAt,
|
|
updatedAt: manga.updatedAt,
|
|
};
|
|
}
|
|
|
|
async updateManga(id: string, data: UpdateMangaDto): Promise<Manga> {
|
|
// Validate input
|
|
this.validateMangaData(data);
|
|
|
|
const updateData = { ...data };
|
|
if (updateData.status) {
|
|
updateData.status = updateData.status.toUpperCase() as any;
|
|
}
|
|
|
|
const manga = await this.prisma.manga.update({
|
|
where: { id },
|
|
data: updateData,
|
|
});
|
|
|
|
return {
|
|
...manga,
|
|
status: manga.status as unknown as MangaStatus,
|
|
dateAdded: manga.dateAdded,
|
|
dateStarted: manga.dateStarted || undefined,
|
|
dateCompleted: manga.dateCompleted || undefined,
|
|
dateFinished: manga.dateFinished || undefined,
|
|
tags: manga.tags ?? [],
|
|
links: manga.links ?? [],
|
|
createdAt: manga.createdAt,
|
|
updatedAt: manga.updatedAt,
|
|
};
|
|
}
|
|
|
|
async deleteManga(id: string): Promise<void> {
|
|
await this.prisma.manga.delete({
|
|
where: { id },
|
|
});
|
|
}
|
|
}
|