From 6cdd9bcf52867b6ccebfed9a5443d3ee98ff45f3 Mon Sep 17 00:00:00 2001 From: Sachin Rawani Date: Wed, 24 Jan 2024 23:16:17 +0530 Subject: [PATCH] Added support for Calibre Recipe --- src/Modules/ConversionModule.ts | 8 ++- src/Modules/ImportationModule.ts | 8 ++- src/Protocols/SetupInputProtocol.ts | 2 +- src/Tools/Converters/RecipeConverterTool.ts | 77 +++++++++++++++++++++ src/Tools/Importers/RecipeImporterTool.ts | 16 +++++ 5 files changed, 104 insertions(+), 7 deletions(-) create mode 100644 src/Tools/Converters/RecipeConverterTool.ts create mode 100644 src/Tools/Importers/RecipeImporterTool.ts diff --git a/src/Modules/ConversionModule.ts b/src/Modules/ConversionModule.ts index e83aa27..4d7c127 100644 --- a/src/Modules/ConversionModule.ts +++ b/src/Modules/ConversionModule.ts @@ -6,18 +6,20 @@ import { ConverterContract } from "@/Protocols/ConverterProtocol" import RSSConverterTool from "@/Tools/Converters/RSSConverterTool" import MangaConverterTool from "@/Tools/Converters/MangaConverterTool" +import RecipeConverterTool from "@/Tools/Converters/RecipeConverterTool" class ConversionModule { - async convert (content: Content): Promise { + async convert(content: Content): Promise { const converter = this.getConverterBySourceConfig(content.sourceConfig) return await converter.convert(content) } - private getConverterBySourceConfig (sourceConfig: SourceConfig): ConverterContract { + private getConverterBySourceConfig(sourceConfig: SourceConfig): ConverterContract { const converterMap: Record> = { rss: RSSConverterTool, - manga: MangaConverterTool + manga: MangaConverterTool, + recipe: RecipeConverterTool } return converterMap[sourceConfig.type] diff --git a/src/Modules/ImportationModule.ts b/src/Modules/ImportationModule.ts index 0412fdf..9ddf609 100644 --- a/src/Modules/ImportationModule.ts +++ b/src/Modules/ImportationModule.ts @@ -3,18 +3,20 @@ import { SourceConfig } from "@/Protocols/SetupInputProtocol" import RSSImporterTool from "@/Tools/Importers/RSSImporterTool" import MangaImporterTool from "@/Tools/Importers/MangaImporterTool" +import RecipeImporterTool from "@/Tools/Importers/RecipeImporterTool" class ImportationModule { - async import (sourceConfig: SourceConfig): Promise> { + async import(sourceConfig: SourceConfig): Promise> { const importer = this.getImporterBySourceConfig(sourceConfig) return await importer.import(sourceConfig) } - private getImporterBySourceConfig (sourceConfig: SourceConfig): ImporterContract { + private getImporterBySourceConfig(sourceConfig: SourceConfig): ImporterContract { const importerMap: Record> = { rss: RSSImporterTool, - manga: MangaImporterTool + manga: MangaImporterTool, + recipe: RecipeImporterTool, } return importerMap[sourceConfig.type] diff --git a/src/Protocols/SetupInputProtocol.ts b/src/Protocols/SetupInputProtocol.ts index e0af3c0..435a5ee 100644 --- a/src/Protocols/SetupInputProtocol.ts +++ b/src/Protocols/SetupInputProtocol.ts @@ -27,7 +27,7 @@ export type SourceConfig = { * Every RSS post will be a new document inside your kindle. */ splitRSSPosts?: boolean - type: "rss" | "manga" + type: "rss" | "manga" | "recipe" } export type KindleConfig = { diff --git a/src/Tools/Converters/RecipeConverterTool.ts b/src/Tools/Converters/RecipeConverterTool.ts new file mode 100644 index 0000000..3b87477 --- /dev/null +++ b/src/Tools/Converters/RecipeConverterTool.ts @@ -0,0 +1,77 @@ +import fs from "fs" + +import { DocumentModel } from "@/Models/DocumentModel" + +import { ConverterContract } from "@/Protocols/ConverterProtocol" +import { Content } from "@/Protocols/ImporterProtocol" +import { GenerateEPUBOptions, EpubContent } from "@/Protocols/EbookGeneratorProtocol" +import { SourceConfig } from "@/Protocols/SetupInputProtocol" + + +import EbookGeneratorService from "@/Services/EbookGeneratorService" +import QueueService from "@/Services/QueueService" + +import FileUtil from "@/Utils/FileUtil" +import DateUtil from "@/Utils/DateUtil" + +class RecipeConverterTool implements ConverterContract { + private readonly queueService = new QueueService({ concurrency: 10 }) + private readonly ebookGeneratorService = new EbookGeneratorService() + + async convert(content: Content): Promise { + const EPUBConfigs = await this.RecipeToEPUBConfig(content.sourceConfig) + const epubFilePath = content.sourceConfig.name + const documents: DocumentModel[] = await Promise.all( + EPUBConfigs.map(async EPUBConfig => ( + await this.queueService.enqueue(async () => { + + const mobiFilePath = await this.RecipeToMOBI(epubFilePath) + + const { filename } = FileUtil.parseFilePath(mobiFilePath) + + const mobiData = fs.createReadStream(mobiFilePath) + + return new DocumentModel({ + title: EPUBConfig.title, + filename, + data: mobiData, + type: content.sourceConfig.type + }) + }) + )) + ) + + return documents + } + + private async RecipeToEPUBConfig(sourceConfig: SourceConfig): Promise { + const metadataTitle = sourceConfig.name + const metadataSubTitle = DateUtil.todayFormattedDate + const content_data: EpubContent = { + title: "", + author: "", + data: "" + } + + return [{ + title: `${metadataTitle} ${metadataSubTitle}`, + author: null, + publisher: null, + cover: null, + content: [content_data], + metadata: { + title: metadataTitle, + subTitle: metadataSubTitle + } + }] + + } + + private async RecipeToMOBI(epubFilePath: string): Promise { + const recipe_name = `${epubFilePath}.recipe` + const mobiFilePath = await this.ebookGeneratorService.generateMOBIFromEPUB(recipe_name) + return mobiFilePath + } +} + +export default new RecipeConverterTool() diff --git a/src/Tools/Importers/RecipeImporterTool.ts b/src/Tools/Importers/RecipeImporterTool.ts new file mode 100644 index 0000000..adbd578 --- /dev/null +++ b/src/Tools/Importers/RecipeImporterTool.ts @@ -0,0 +1,16 @@ +import { Content, ImporterContract } from "@/Protocols/ImporterProtocol" +import { SourceConfig } from "@/Protocols/SetupInputProtocol" + + +class RecipeImporterTool implements ImporterContract { + + async import(sourceConfig: SourceConfig): Promise> { + + return { + data: null, + sourceConfig + } + } +} + +export default new RecipeImporterTool()