[HentaiVn.buzz + YuriGarden] Fixes (#1968)

* [HentaiVn.buzz] Fix "Nothing found" errors

* [YuriGarden] Add author search support + Small fixes
master
Draken 10 months ago committed by GitHub
parent 66f25c30fc
commit 5d9b56296b
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

@ -1,157 +1,113 @@
package org.koitharu.kotatsu.parsers.site.vi package org.koitharu.kotatsu.parsers.site.vi
import org.jsoup.nodes.Document
import org.koitharu.kotatsu.parsers.MangaLoaderContext import org.koitharu.kotatsu.parsers.MangaLoaderContext
import org.koitharu.kotatsu.parsers.MangaSourceParser import org.koitharu.kotatsu.parsers.MangaSourceParser
import org.koitharu.kotatsu.parsers.config.ConfigKey import org.koitharu.kotatsu.parsers.config.ConfigKey
import org.koitharu.kotatsu.parsers.core.LegacyPagedMangaParser import org.koitharu.kotatsu.parsers.core.LegacyPagedMangaParser
import org.koitharu.kotatsu.parsers.model.* import org.koitharu.kotatsu.parsers.model.*
import org.koitharu.kotatsu.parsers.network.UserAgents
import org.koitharu.kotatsu.parsers.util.* import org.koitharu.kotatsu.parsers.util.*
import java.util.* import java.util.*
@MangaSourceParser("HENTAIVNBUZZ", "HentaiVn.buzz", "vi", type = ContentType.HENTAI) @MangaSourceParser("HENTAIVNBUZZ", "HentaiVn.buzz", "vi", type = ContentType.HENTAI)
internal class HentaiVnBuzz(context: MangaLoaderContext) : internal class HentaiVnBuzz(context: MangaLoaderContext) :
LegacyPagedMangaParser(context, MangaParserSource.HENTAIVNBUZZ, 24) { LegacyPagedMangaParser(context, MangaParserSource.HENTAIVNBUZZ, 60) {
override val configKeyDomain = ConfigKey.Domain("hentaivn.email") override val configKeyDomain = ConfigKey.Domain("hentaivn.beer")
override fun onCreateConfig(keys: MutableCollection<ConfigKey<*>>) { override val availableSortOrders: Set<SortOrder> =
super.onCreateConfig(keys) EnumSet.of(
keys.add(userAgentKey) SortOrder.UPDATED,
}
override val userAgentKey = ConfigKey.UserAgent(UserAgents.CHROME_DESKTOP)
override val availableSortOrders: Set<SortOrder> = EnumSet.of(
SortOrder.NEWEST, SortOrder.NEWEST,
SortOrder.POPULARITY, SortOrder.POPULARITY,
SortOrder.UPDATED, SortOrder.RATING,
) )
override val filterCapabilities: MangaListFilterCapabilities override val filterCapabilities: MangaListFilterCapabilities
get() = MangaListFilterCapabilities( get() = MangaListFilterCapabilities(
isMultipleTagsSupported = true,
isTagsExclusionSupported = true,
isSearchSupported = true, isSearchSupported = true,
) )
override suspend fun getFilterOptions() = MangaListFilterOptions( override suspend fun getFilterOptions() = MangaListFilterOptions(
availableTags = fetchTags(), availableTags = fetchAvailableTags(),
availableStates = EnumSet.of(MangaState.ONGOING, MangaState.FINISHED), availableStates = EnumSet.of(MangaState.ONGOING, MangaState.FINISHED),
) )
override suspend fun getListPage(page: Int, order: SortOrder, filter: MangaListFilter): List<Manga> { override suspend fun getListPage(page: Int, order: SortOrder, filter: MangaListFilter): List<Manga> {
val url = when { val url = buildString {
!filter.query.isNullOrEmpty() -> { append("https://")
buildString { append(domain)
append("/tim-kiem?key_word=") append("/tim-kiem-nang-cao")
append(filter.query.urlEncoded())
if (page > 1) {
append("&page=")
append(page)
}
}
}
filter.tags.isNotEmpty() -> { append("?page=")
val tag = filter.tags.first()
buildString {
append("/the-loai/")
append(tag.key)
append("?")
when (order) {
SortOrder.NEWEST -> append("sort=0")
SortOrder.UPDATED -> append("sort=1")
SortOrder.POPULARITY -> append("sort=2")
else -> append("sort=0")
}
if (filter.states.isNotEmpty()) {
filter.states.forEach {
when (it) {
MangaState.ONGOING -> append("&is_full=0")
MangaState.FINISHED -> append("&is_full=1")
else -> append("")
}
}
}
if (page > 1) {
append("&page=")
append(page) append(page)
}
}
}
else -> { append("&sort=")
buildString { append(
append("/danh-sach/truyen-moi?")
when (order) { when (order) {
SortOrder.NEWEST -> append("sort=0") SortOrder.UPDATED -> "0"
SortOrder.UPDATED -> append("sort=1") SortOrder.NEWEST -> "1"
SortOrder.POPULARITY -> append("sort=2") SortOrder.POPULARITY -> "2"
else -> append("sort=0") SortOrder.RATING -> "6"
} else -> "0"
if (filter.states.isNotEmpty()) { },
filter.states.forEach { )
when (it) {
MangaState.ONGOING -> append("&is_full=0") append("&status=")
MangaState.FINISHED -> append("&is_full=1") append(
else -> append("") when (filter.states.oneOrThrowIfMany()) {
} MangaState.ONGOING -> "1"
} MangaState.FINISHED -> "2"
} else -> "0"
if (page > 1) { },
append("&page=") )
append(page)
} if (filter.tags.isNotEmpty()) {
} append("&category=")
append(filter.tags.joinToString(",") { it.key })
} }
if (filter.tagsExclude.isNotEmpty()) {
append("&notcategory=")
append(filter.tagsExclude.joinToString(",") { it.key })
} }
val fullUrl = url.toAbsoluteUrl(domain) if (!filter.query.isNullOrEmpty()) {
val doc = webClient.httpGet(fullUrl).parseHtml() clear()
return when {
!filter.query.isNullOrEmpty() -> parseSearchManga(doc) append("https://")
filter.tags.isNotEmpty() -> parseSearchManga(doc) append(domain)
else -> parseListManga(doc) append("/tim-kiem?q=")
append(filter.query.urlEncoded())
return@buildString // end of buildString
} }
} }
private fun parseSearchManga(doc: Document): List<Manga> { val doc = webClient.httpGet(url).parseHtml()
return doc.select(".story-item-list.d-flex.align-items-center.position-relative.mb-1").map { div -> return doc.select("ul.list_grid.grid > li").map { element ->
val href = div.selectFirstOrThrow("a.story-item-list__image").attrAsRelativeUrl("href") val aTag = element.selectFirstOrThrow("h3 a")
val coverUrl = div.selectFirst("img")?.attr("data-src") val tags = element.select(".genre-item").mapToSet {
val title = div.selectFirst("img")?.attr("alt").orEmpty() MangaTag(
Manga( key = it.attr("href").substringAfterLast('-').substringBeforeLast('.'),
id = generateUid(href), title = it.text().toTitleCase(sourceLocale),
title = title,
altTitles = emptySet(),
url = href,
publicUrl = href.toAbsoluteUrl(domain),
rating = RATING_UNKNOWN,
contentRating = if (isNsfwSource) ContentRating.ADULT else null,
coverUrl = coverUrl,
tags = emptySet(),
state = null,
authors = emptySet(),
source = source, source = source,
) )
} }
}
private fun parseListManga(doc: Document): List<Manga> { val href = aTag.attrAsRelativeUrl("href")
return doc.select(".story-item-list.d-flex.align-items-center.position-relative.mb-1").map { div ->
val href = div.selectFirstOrThrow("a.story-item-list__image").attrAsRelativeUrl("href")
val coverUrl = div.selectFirst("img")?.attr("data-src").orEmpty()
val title = div.selectFirst("img")?.attr("alt").orEmpty()
Manga( Manga(
id = generateUid(href), id = generateUid(href),
title = title, title = aTag.text(),
altTitles = emptySet(), altTitles = emptySet(),
url = href, url = href,
publicUrl = href.toAbsoluteUrl(domain), publicUrl = aTag.attrAsAbsoluteUrl("href"),
rating = RATING_UNKNOWN, rating = RATING_UNKNOWN,
contentRating = if (isNsfwSource) ContentRating.ADULT else null, contentRating = ContentRating.ADULT,
coverUrl = coverUrl, coverUrl = element.selectFirst(".book_avatar a img")?.src(),
tags = emptySet(), tags = tags,
state = null, state = null,
authors = emptySet(), authors = emptySet(),
source = source, source = source,
@ -161,26 +117,29 @@ internal class HentaiVnBuzz(context: MangaLoaderContext) :
override suspend fun getDetails(manga: Manga): Manga { override suspend fun getDetails(manga: Manga): Manga {
val doc = webClient.httpGet(manga.url.toAbsoluteUrl(domain)).parseHtml() val doc = webClient.httpGet(manga.url.toAbsoluteUrl(domain)).parseHtml()
val author = doc.select("p:contains(Tác giả:) a").text().nullIfEmpty() val tags = doc.select("ul.list01 li").mapToSet {
return manga.copy(
authors = setOfNotNull(author),
tags = doc.select("div.mb-1 span a").mapToSet { element ->
MangaTag( MangaTag(
key = element.attr("href").substringAfter("/the-loai/"), key = it.attr("href").substringAfterLast('-').substringBeforeLast('.'),
title = element.text().substringBefore(',').trim(), // force trim before , symbol and space title = it.text().toTitleCase(sourceLocale),
source = source, source = source,
) )
}, }
description = null, val author = doc.selectFirst("li.author a")?.textOrNull()
state = when (doc.select("p:contains(Trạng thái:) span").text()) {
"Đang ra" -> MangaState.ONGOING return manga.copy(
altTitles = setOfNotNull(doc.selectFirst("h2.other-name")?.textOrNull()),
authors = setOfNotNull(author),
tags = tags,
description = doc.selectFirst("div.story-detail-info")?.html(),
state = when (doc.selectFirst(".status p.col-xs-9")?.text()) {
"Đang tiến hành" -> MangaState.ONGOING
"Hoàn thành" -> MangaState.FINISHED "Hoàn thành" -> MangaState.FINISHED
else -> null else -> null
}, },
chapters = doc.select("div.story-detail__list-chapter--list ul.list-unstyled li a") chapters = doc.select("div.list_chapter div.works-chapter-item").mapChapters(reversed = true) { i, div ->
.mapIndexed { i, element -> val a = div.selectFirstOrThrow("a")
val href = element.attrAsRelativeUrl("href") val href = a.attrAsRelativeUrl("href")
val name = element.text().removePrefix("- ") val name = a.text()
MangaChapter( MangaChapter(
id = generateUid(href), id = generateUid(href),
title = name, title = name,
@ -188,7 +147,7 @@ internal class HentaiVnBuzz(context: MangaLoaderContext) :
volume = 0, volume = 0,
url = href, url = href,
scanlator = null, scanlator = null,
uploadDate = 0, uploadDate = 0L,
branch = null, branch = null,
source = source, source = source,
) )
@ -199,9 +158,8 @@ internal class HentaiVnBuzz(context: MangaLoaderContext) :
override suspend fun getPages(chapter: MangaChapter): List<MangaPage> { override suspend fun getPages(chapter: MangaChapter): List<MangaPage> {
val fullUrl = chapter.url.toAbsoluteUrl(domain) val fullUrl = chapter.url.toAbsoluteUrl(domain)
val doc = webClient.httpGet(fullUrl).parseHtml() val doc = webClient.httpGet(fullUrl).parseHtml()
val imageUrls = doc.select("meta[property='og:image']").map { it.attr("content") } return doc.select(".chapter_content img").map { img ->
val finalUrls = imageUrls.drop(1) val url = img.requireSrc()
return finalUrls.map { url ->
MangaPage( MangaPage(
id = generateUid(url), id = generateUid(url),
url = url, url = url,
@ -211,18 +169,15 @@ internal class HentaiVnBuzz(context: MangaLoaderContext) :
} }
} }
private suspend fun fetchTags(): Set<MangaTag> { private suspend fun fetchAvailableTags(): Set<MangaTag> {
val doc = webClient.httpGet("https://$domain/").parseHtml() val doc = webClient.httpGet("https://$domain/tim-kiem-nang-cao").parseHtml()
val list = doc.select("ul.dropdown-menu.dropdown-menu-custom li a") val elements = doc.select(".genre-item")
return list.mapToSet { tags -> return elements.mapIndexed { i, element ->
val href = tags.attr("href")
val key = href.substringAfter("/the-loai/").substringBefore("/")
val title = tags.text()
MangaTag( MangaTag(
key = key, key = (i + 1).toString(),
title = title, title = element.text().toTitleCase(sourceLocale),
source = source, source = source,
) )
} }.toSet()
} }
} }

@ -42,6 +42,7 @@ internal abstract class YuriGardenParser(
isSearchSupported = true, isSearchSupported = true,
isMultipleTagsSupported = true, isMultipleTagsSupported = true,
isSearchWithFiltersSupported = true, isSearchWithFiltersSupported = true,
isAuthorSearchSupported = true,
) )
override suspend fun getFilterOptions(): MangaListFilterOptions { override suspend fun getFilterOptions(): MangaListFilterOptions {
@ -99,6 +100,19 @@ internal abstract class YuriGardenParser(
append("&genre=") append("&genre=")
append(filter.tags.joinToString(separator = ",") { it.key }) append(filter.tags.joinToString(separator = ",") { it.key })
} }
if (!filter.author.isNullOrEmpty()) {
clear()
append("https://")
append(apiSuffix)
append("/creators/authors/")
append(
filter.author.substringAfter("(").substringBefore(")")
)
return@buildString // end of buildString
}
} }
val json = webClient.httpGet(url).parseJson() val json = webClient.httpGet(url).parseJson()
@ -106,8 +120,10 @@ internal abstract class YuriGardenParser(
return data.mapJSON { jo -> return data.mapJSON { jo ->
val id = jo.getLong("id") val id = jo.getLong("id")
val allTags = fetchTags().orEmpty() val altTitles = setOf(jo.optString("anotherName", null))
val tags = allTags.let { allTags -> .filterNotNull()
.toSet()
val tags = fetchTags().let { allTags ->
jo.optJSONArray("genres")?.asTypedList<String>()?.mapNotNullToSet { g -> jo.optJSONArray("genres")?.asTypedList<String>()?.mapNotNullToSet { g ->
allTags.find { x -> x.key == g } allTags.find { x -> x.key == g }
} }
@ -118,12 +134,12 @@ internal abstract class YuriGardenParser(
url = "/comics/$id", url = "/comics/$id",
publicUrl = "https://$domain/comic/$id", publicUrl = "https://$domain/comic/$id",
title = jo.getString("title"), title = jo.getString("title"),
altTitles = setOf(jo.getString("anotherName")), altTitles = altTitles,
coverUrl = jo.getString("thumbnail"), coverUrl = jo.getString("thumbnail"),
largeCoverUrl = jo.getString("thumbnail"), largeCoverUrl = jo.getString("thumbnail"),
authors = emptySet(), authors = emptySet(),
tags = tags, tags = tags,
state = when(jo.getString("status")) { state = when(jo.optString("status")) {
"ongoing" -> MangaState.ONGOING "ongoing" -> MangaState.ONGOING
"completed" -> MangaState.FINISHED "completed" -> MangaState.FINISHED
"hiatus" -> MangaState.PAUSED "hiatus" -> MangaState.PAUSED
@ -131,7 +147,7 @@ internal abstract class YuriGardenParser(
"oncoming" -> MangaState.UPCOMING "oncoming" -> MangaState.UPCOMING
else -> null else -> null
}, },
description = jo.getString("description"), description = jo.optString("description").orEmpty(),
contentRating = if (jo.getBooleanOrDefault("r18", false)) ContentRating.ADULT else ContentRating.SUGGESTIVE, contentRating = if (jo.getBooleanOrDefault("r18", false)) ContentRating.ADULT else ContentRating.SUGGESTIVE,
source = source, source = source,
rating = RATING_UNKNOWN, rating = RATING_UNKNOWN,
@ -144,9 +160,11 @@ internal abstract class YuriGardenParser(
val json = webClient.httpGet("https://$apiSuffix/comics/${id}").parseJson() val json = webClient.httpGet("https://$apiSuffix/comics/${id}").parseJson()
val authors = json.optJSONArray("authors")?.mapJSONToSet { jo -> val authors = json.optJSONArray("authors")?.mapJSONToSet { jo ->
jo.getString("name") jo.getString("name") + " (${jo.getLong("id")})"
}.orEmpty() }.orEmpty()
val altTitles = setOf(json.getString("anotherName"))
val description = json.getString("description")
val team = json.optJSONArray("teams")?.getJSONObject(0)?.getString("name") val team = json.optJSONArray("teams")?.getJSONObject(0)?.getString("name")
val chaptersDeferred = async { val chaptersDeferred = async {
@ -154,6 +172,7 @@ internal abstract class YuriGardenParser(
} }
manga.copy( manga.copy(
altTitles = altTitles,
authors = authors, authors = authors,
chapters = chaptersDeferred.await().mapChapters() { _, jo -> chapters = chaptersDeferred.await().mapChapters() { _, jo ->
val chapId = jo.getLong("id") val chapId = jo.getLong("id")
@ -168,7 +187,8 @@ internal abstract class YuriGardenParser(
branch = null, branch = null,
source = source, source = source,
) )
} },
description = description,
) )
} }

Loading…
Cancel
Save