Skip to content

Commit

Permalink
Merge branch 'dev' into TokyoTF
Browse files Browse the repository at this point in the history
  • Loading branch information
koikiss-dev authored Feb 2, 2024
2 parents 21b5c14 + 4acaac3 commit 0adc3e5
Show file tree
Hide file tree
Showing 8 changed files with 472 additions and 0 deletions.
2 changes: 2 additions & 0 deletions src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ import comick from "@routes/manga/comick/ComickRoutes";
import inmanga from "@routes/manga/inmanga/InmangaRoutes";
import nhentai from "@routes/manga/nhentai/NhentaiRoutes"
import mangareader from "@routes/manga/mangareader/MangaReaderRoutes";
import manganelo from "@routes/manga/manganelo/ManganeloRoutes";

const app = express();
const port = process.env.PORT || 3000;
Expand Down Expand Up @@ -56,6 +57,7 @@ app.use(comick);
app.use(inmanga);
app.use(nhentai)
app.use(mangareader);
app.use(manganelo);

/* Manga */

Expand Down
33 changes: 33 additions & 0 deletions src/routes/v1/manga/manganelo/ManganeloRoutes.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
import { manganatoOrderByOptionsList } from "@providers/manganelo/ManganatoTypes";
import { Router } from "express";
import { Manganelo } from "../../../../scraper/sites/manga/manganelo/Manganelo";

const router = Router();
const manganelo = new Manganelo();

router.get(`/manga/${manganelo.name}/title/:id`, async (req, res) => {
const result = await manganelo.GetMangaInfo(
req.params.id as unknown as string,
);

return res.status(200).send(result);
});

router.get(`/manga/${manganelo.name}/filter`, async (req, res) => {
const result = await manganelo.Filter({
sts: req.query.status as unknown as "ongoing" | "completed",
genres: req.query.genres as unknown as string,
orby: req.query.order as unknown as typeof manganatoOrderByOptionsList[number],
page: req.query.page as unknown as number
});

return res.status(200).send(result);
})

router.get(`/manga/${manganelo.name}/chapter/:id`, async (req, res) => {
const result = await manganelo.GetMangaChapters(req.params.id as unknown as string, req.query.num as unknown as number);

return res.status(200).send(result);
});

export default router;
14 changes: 14 additions & 0 deletions src/scraper/sites/manga/manganelo/ManganatoManagerUtils.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
import { ManganatoAdvancedSearchURLManager } from "./managers/ManganatoURLManager";

export class ManganatoManagerUtils {
private static instance: ManganatoManagerUtils;
readonly url: ManganatoAdvancedSearchURLManager = new ManganatoAdvancedSearchURLManager();

private constructor() { }

static get Instance() {
if (!ManganatoManagerUtils.instance) ManganatoManagerUtils.instance = new ManganatoManagerUtils();

return ManganatoManagerUtils.instance;
}
}
63 changes: 63 additions & 0 deletions src/scraper/sites/manga/manganelo/ManganatoTypes.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
export const manganatoOrderByOptionsList = ["topview", "newest", "az"] as const;
export type manganatoOrderByOptions = typeof manganatoOrderByOptionsList[number];

export interface IManganatoFilterParams {
/**
* Manga status
*
* Available status: "ongoing", "completed", empty string | null (for both)
*/
sts: "ongoing" | "completed";
/** Order by */
orby: manganatoOrderByOptions;
genres: string;
/** Results page */
page: number;
};

export type ManganatoFilterURLParams = keyof Omit<IManganatoFilterParams, "genres"> | "g_i" | "s";

export const manganatoGenreList = {
action: 2,
adult: 3,
adventure: 4,
comedy: 6,
cooking: 7,
doujinshi: 9,
drama: 10,
ecchi: 11,
fantasy: 12,
genderbender: 13,
harem: 14,
historical: 15,
horror: 16,
josei: 17,
martialarts: 19,
mature: 20,
mecha: 21,
medical: 22,
mystery: 24,
oneshot: 25,
psychological: 26,
romance: 27,
schoollife: 28,
scifi: 29,
seinen: 30,
shoujo: 31,
shoujoai: 32,
shounen: 33,
shounenai: 34,
sliceoflife: 35,
smut: 36,
sports: 37,
supernatural: 38,
tragedy: 39,
webtoons: 40,
yaoi: 41,
yuri: 42,
manhwa: 43,
manhua: 44,
isekai: 45,
pornographic: 47,
erotica: 48
} as const;
163 changes: 163 additions & 0 deletions src/scraper/sites/manga/manganelo/Manganelo.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,163 @@
import { IMangaResult, Manga, MangaChapter } from "../../../../types/manga";
import axios from "axios";
import { load } from "cheerio";
import { Image } from "../../../../types/image";
import { ManganatoManagerUtils } from "./ManganatoManagerUtils";
import { IManganatoFilterParams } from "./ManganatoTypes";
import { ResultSearch } from "../../../../types/search";

export class Manganelo {
private readonly url = "https://manganelo.tv"; //chapmanganelo.com //mangakakalot.tv;
readonly name = "manganelo";
private readonly manager = ManganatoManagerUtils.Instance;

private GetMangaDescription(data: cheerio.Root) {
if (
data("div#panel-story-info-description").length == 0 &&
data("div#panel-story-info-description h3").length == 0
)
return null;

data("div#panel-story-info-description h3").remove();

return data("div#panel-story-info-description").text().trim();
}

private GetMangaStatus(data: cheerio.Root) {
const selector = data("div.panel-story-info > div.story-info-right > table > tbody > tr:nth-child(3) > td.table-value");

if (selector.length == 0)
return null;

if (selector.text().trim() == "Ongoing")
return "ongoing";
else
return "completed";
}

private GetMangaAuthors(data: cheerio.Root): string[] | null {
const selector = data("div.panel-story-info > div.story-info-right > table > tbody > tr:nth-child(2) > td.table-value");

if (selector.length == 0 && selector.find("a.a-h").length == 0)
return null;

return selector.find("a.a-h").map((_, element) => {
return data(element).text().trim();
}).get();
}

private GetMangaGenres(data: cheerio.Root): string[] | null {
const selector = data("div.panel-story-info > div.story-info-right > table > tbody > tr:nth-child(4) > td.table-value");

if (selector.length == 0 && selector.find("a.a-h").length == 0)
return null;

return selector.find("a.a-h").map((_, element) => {
return data(element).text().trim();
}).get();
}

private isNsfw(genres: string[]) {
return genres.some(genre => genre === "Pornographic" || genre === "Mature" || genre === "Erotica");
}

private GetMangaPages(data: cheerio.Root) {
if (data("div.container-chapter-reader").length == 0 && data("div.container-chapter-reader > img").length == 0)
return null;

return data("div.container-chapter-reader > img").map((_, element) => data(element).attr("data-src")).get();
}

private GetMangaSearchResults(data: cheerio.Root): IMangaResult[] | null {
const section = data("div.panel-content-genres");
if (section.length === 0)
return null;

return section.find("div.content-genres-item").map((_, element) => {
const mangaResultId = data(element).find("a.genres-item-img").attr("href").split("-").at(-1);
const name = data(element).find("a.genres-item-img").attr("title").trim();

const mangaInfoResults: IMangaResult = {
id: mangaResultId,
title: name,
url: `/manga/${this.name}/title/${mangaResultId}`
}

return mangaInfoResults;
}).get();
}

async GetMangaInfo(mangaId: string) {
const { data } = await axios.get(`${this.url}/manga/manga-${mangaId}`);
const $ = load(data);

const manga = new Manga;

const title = $("div.panel-story-info > div.story-info-right > h1").text().trim();
const description = this.GetMangaDescription($);
const thumbnail = this.url + $("div.panel-story-info > div.story-info-left > span.info-image > img").attr("src");
const altTitle = $("table > tbody > tr:nth-child(1) > td.table-value > h2").text().trim();
const status = this.GetMangaStatus($);
const authors = this.GetMangaAuthors($);
const genres = this.GetMangaGenres($);
const chapters = $("div.panel-story-chapter-list").find("ul > li.a-h").map((_, element) => {
const chapter = new MangaChapter;
const url = $(element).find("a.chapter-name").attr("href");

const chapterId = url.substring(url.lastIndexOf("-") + 1);

chapter.id = Number(chapterId);
chapter.title = $(element).find("a.chapter-name").text().trim();
chapter.url = `/manga/${this.name}/chapter/${mangaId}?num=${chapterId}`;
chapter.number = Number(chapterId);
chapter.images = null;

return chapter;
}).get();

manga.id = mangaId;
manga.url = `/manga/${this.name}/title/${mangaId}`;
manga.title = title;
manga.altTitles = Array.of(altTitle);
manga.thumbnail = new Image(thumbnail);
manga.description = description;
manga.status = status;
manga.authors = authors;
manga.genres = genres;
manga.characters = null;
manga.chapters = chapters;
manga.volumes = null;
manga.isNSFW = this.isNsfw(genres);

return manga;
}

async Filter(params: IManganatoFilterParams) {
const url = this.manager.url.generate(params);

const { data } = await axios.get(url);
const $ = load(data);

const mangaResultSearch = new ResultSearch<IMangaResult>();
mangaResultSearch.results = this.GetMangaSearchResults($);

return mangaResultSearch;
}

async GetMangaChapters(mangaId: string, chapterNumber: number) {
const { data } = await axios.get(`${this.url}/chapter/manga-${mangaId}/chapter-${chapterNumber}`);
const $ = load(data);

const images = this.GetMangaPages($);
const name = $("body > div.body-site > div:nth-child(1) > div.panel-breadcrumb > a").eq(-1).attr("title") || null;
const chapter = new MangaChapter;

chapter.id = Number(chapterNumber);
chapter.title = name;
chapter.url = `/manga/${this.name}/chapter/${mangaId}?num=${chapterNumber}`;
chapter.number = Number(chapterNumber);
chapter.images = images;

return chapter;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
export abstract class ManganatoManager {
abstract generate(item: unknown, ...args: unknown[]): unknown;
}
61 changes: 61 additions & 0 deletions src/scraper/sites/manga/manganelo/managers/ManganatoURLManager.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
import { URLSearchParams } from "url";
import { IManganatoFilterParams, manganatoGenreList, manganatoOrderByOptions, manganatoOrderByOptionsList, ManganatoFilterURLParams } from "../ManganatoTypes";
import { ManganatoManager } from "./ManganatoManager";

export class ManganatoAdvancedSearchURLManager extends ManganatoManager {
private readonly baseURL = "https://manganato.com/advanced_search";
private readonly separator = " ";

private splitGenresToArray(genres: string) {
return genres.split(this.separator);
}

private processGenres(genresArray: string[]): number[] {
let arrGenerated: number[] = [];

for (let genre of genresArray) {
if (manganatoGenreList[genre.toLowerCase()])
arrGenerated.push(manganatoGenreList[genre.toLowerCase()]);
else continue;
}

return arrGenerated;
}

private formatGenres(genresProcessedArray: number[]) {
return `_${genresProcessedArray.join("_")}_`;
}

private processStatus(status: unknown) {
return (typeof status === "string" && (status.toLowerCase() === "ongoing" || status.toLowerCase() === "completed"))
? status
: "";
}

private processOrderBy(order: unknown) {
return (typeof order === "string" && manganatoOrderByOptionsList.includes(order.toLowerCase() as manganatoOrderByOptions))
? order
: "";
}

generate(params: Partial<IManganatoFilterParams>) {
const urlParamsObject = {
s: "all",
g_i: "",
sts: this.processStatus(params.sts),
orby: this.processOrderBy(params.orby),
page: params.page ? params.page.toString() : ""
} satisfies Record<ManganatoFilterURLParams, string>;

if (params.genres) {
const splitted = this.splitGenresToArray(params.genres);
const processed = this.processGenres(splitted);

urlParamsObject.g_i = this.formatGenres(processed);
}

const urlParams = new URLSearchParams(urlParamsObject);

return `${this.baseURL}?${urlParams.toString()}`;
}
}
Loading

0 comments on commit 0adc3e5

Please sign in to comment.