cloudstream-extensions-hexated/KuramanimeProvider/src/main/kotlin/com/hexated/KuramanimeProvider.kt

225 lines
8.5 KiB
Kotlin
Raw Normal View History

2022-09-09 16:09:54 +00:00
package com.hexated
import com.lagradost.cloudstream3.*
import com.lagradost.cloudstream3.LoadResponse.Companion.addAniListId
import com.lagradost.cloudstream3.LoadResponse.Companion.addMalId
2022-09-09 16:09:54 +00:00
import com.lagradost.cloudstream3.utils.ExtractorLink
import com.lagradost.cloudstream3.utils.Qualities
import com.lagradost.cloudstream3.utils.loadExtractor
2022-09-09 16:09:54 +00:00
import org.jsoup.Jsoup
import org.jsoup.nodes.Element
class KuramanimeProvider : MainAPI() {
2023-08-26 23:13:47 +00:00
override var mainUrl = "https://kuramanime.pro"
2022-09-09 16:09:54 +00:00
override var name = "Kuramanime"
override val hasQuickSearch = false
override val hasMainPage = true
override var lang = "id"
override val hasDownloadSupport = true
2023-09-23 22:05:33 +00:00
private var headers: Map<String,String> = mapOf()
private var cookies: Map<String,String> = mapOf()
2022-09-09 16:09:54 +00:00
override val supportedTypes = setOf(
TvType.Anime,
TvType.AnimeMovie,
TvType.OVA
)
companion object {
2023-04-04 00:44:24 +00:00
fun getType(t: String, s: Int): TvType {
return if (t.contains("OVA", true) || t.contains("Special")) TvType.OVA
2023-04-04 00:44:24 +00:00
else if (t.contains("Movie", true) && s == 1) TvType.AnimeMovie
else TvType.Anime
}
2022-09-09 16:09:54 +00:00
fun getStatus(t: String): ShowStatus {
return when (t) {
"Selesai Tayang" -> ShowStatus.Completed
"Sedang Tayang" -> ShowStatus.Ongoing
else -> ShowStatus.Completed
}
}
}
override val mainPage = mainPageOf(
"$mainUrl/anime/ongoing?order_by=updated&page=" to "Sedang Tayang",
"$mainUrl/anime/finished?order_by=updated&page=" to "Selesai Tayang",
"$mainUrl/properties/season/summer-2022?order_by=most_viewed&page=" to "Dilihat Terbanyak Musim Ini",
"$mainUrl/anime/movie?order_by=updated&page=" to "Film Layar Lebar",
)
override suspend fun getMainPage(
page: Int,
request: MainPageRequest
): HomePageResponse {
val document = app.get(request.data + page).document
val home = document.select("div.col-lg-4.col-md-6.col-sm-6").mapNotNull {
it.toSearchResult()
}
return newHomePageResponse(request.name, home)
}
private fun getProperAnimeLink(uri: String): String {
return if (uri.contains("/episode")) {
Regex("(.*)/episode/.+").find(uri)?.groupValues?.get(1).toString() + "/"
} else {
uri
}
}
private fun Element.toSearchResult(): AnimeSearchResponse? {
val href = getProperAnimeLink(fixUrl(this.selectFirst("a")!!.attr("href")))
val title = this.selectFirst("h5 a")?.text() ?: return null
val posterUrl = fixUrl(this.select("div.product__item__pic.set-bg").attr("data-setbg"))
2023-01-14 09:40:35 +00:00
val episode = this.select("div.ep span").text().let {
2023-01-23 23:21:04 +00:00
Regex("Ep\\s(\\d+)\\s/").find(it)?.groupValues?.getOrNull(1)?.toIntOrNull()
2023-01-14 09:40:35 +00:00
}
2022-09-09 16:09:54 +00:00
return newAnimeSearchResponse(title, href, TvType.Anime) {
this.posterUrl = posterUrl
addSub(episode)
}
}
override suspend fun search(query: String): List<SearchResponse> {
val link = "$mainUrl/anime?search=$query&order_by=latest"
val document = app.get(link).document
2022-12-29 11:48:41 +00:00
return document.select("div#animeList div.product__item").mapNotNull {
2022-09-09 16:09:54 +00:00
it.toSearchResult()
}
}
override suspend fun load(url: String): LoadResponse {
val document = app.get(url).document
val title = document.selectFirst(".anime__details__title > h3")!!.text().trim()
val poster = document.selectFirst(".anime__details__pic")?.attr("data-setbg")
val tags = document.select("div.anime__details__widget > div > div:nth-child(2) > ul > li:nth-child(1)")
.text().trim().replace("Genre: ", "").split(", ")
2022-09-09 16:09:54 +00:00
2023-01-23 23:21:04 +00:00
val year = Regex("\\D").replace(
2022-09-09 16:09:54 +00:00
document.select("div.anime__details__widget > div > div:nth-child(1) > ul > li:nth-child(5)")
.text().trim().replace("Musim: ", ""), ""
).toIntOrNull()
val status = getStatus(
document.select("div.anime__details__widget > div > div:nth-child(1) > ul > li:nth-child(3)")
.text().trim().replace("Status: ", "")
)
val description = document.select(".anime__details__text > p").text().trim()
2022-12-29 11:48:41 +00:00
val episodes = mutableListOf<Episode>()
2023-01-23 09:36:47 +00:00
for (i in 1..6) {
2022-12-29 11:48:41 +00:00
val doc = app.get("$url?page=$i").document
val eps = Jsoup.parse(doc.select("#episodeLists").attr("data-content")).select("a.btn.btn-sm.btn-danger")
.mapNotNull {
val name = it.text().trim()
val episode = Regex("(\\d+[.,]?\\d*)").find(name)?.groupValues?.getOrNull(0)
?.toIntOrNull()
val link = it.attr("href")
Episode(link, episode = episode)
}
2022-12-29 11:48:41 +00:00
if(eps.isEmpty()) break else episodes.addAll(eps)
}
2022-09-09 16:09:54 +00:00
val type = getType(document.selectFirst("div.col-lg-6.col-md-6 ul li:contains(Tipe:) a")?.text()?.lowercase() ?: "tv", episodes.size)
2022-09-09 16:09:54 +00:00
val recommendations = document.select("div#randomList > a").mapNotNull {
val epHref = it.attr("href")
val epTitle = it.select("h5.sidebar-title-h5.px-2.py-2").text()
val epPoster = it.select(".product__sidebar__view__item.set-bg").attr("data-setbg")
newAnimeSearchResponse(epTitle, epHref, TvType.Anime) {
this.posterUrl = epPoster
addDubStatus(dubExist = false, subExist = true)
}
}
val tracker = APIHolder.getTracker(listOf(title),TrackerType.getTypes(type),year,true)
return newAnimeLoadResponse(title, url, type) {
2022-09-09 16:09:54 +00:00
engName = title
posterUrl = tracker?.image ?: poster
backgroundPosterUrl = tracker?.cover
2022-09-09 16:09:54 +00:00
this.year = year
addEpisodes(DubStatus.Subbed, episodes)
showStatus = status
plot = description
this.tags = tags
this.recommendations = recommendations
addMalId(tracker?.malId)
addAniListId(tracker?.aniId?.toIntOrNull())
2022-09-09 16:09:54 +00:00
}
}
private suspend fun invokeLocalSource(
url: String,
2023-05-27 18:15:54 +00:00
server: String,
ref: String,
callback: (ExtractorLink) -> Unit
) {
val document = app.get(
url,
referer = ref,
2023-09-23 22:05:33 +00:00
headers = headers,
cookies = cookies
).document
document.select("video#player > source").map {
val link = fixUrl(it.attr("src"))
val quality = it.attr("size").toIntOrNull()
callback.invoke(
ExtractorLink(
2023-05-27 18:15:54 +00:00
fixTitle(server),
fixTitle(server),
link,
2023-09-23 22:05:33 +00:00
referer = "",
quality = quality ?: Qualities.Unknown.value,
2023-10-27 12:46:01 +00:00
// headers = mapOf(
// "Accept" to "video/webm,video/ogg,video/*;q=0.9,application/ogg;q=0.7,audio/*;q=0.6,*/*;q=0.5",
// "Range" to "bytes=0-",
// "Sec-Fetch-Dest" to "video",
// "Sec-Fetch-Mode" to "no-cors",
// ),
)
)
}
}
2022-09-09 16:09:54 +00:00
override suspend fun loadLinks(
data: String,
isCasting: Boolean,
subtitleCallback: (SubtitleFile) -> Unit,
callback: (ExtractorLink) -> Unit
): Boolean {
2023-09-23 22:05:33 +00:00
val req = app.get(data)
val res = req.document
val token = res.select("meta[name=csrf-token]").attr("content")
2023-11-01 04:31:17 +00:00
val st = res.select("input#kuramaRoute").attr("value")
2023-09-23 22:05:33 +00:00
headers = mapOf(
"X-Requested-With" to "XMLHttpRequest",
"X-CSRF-TOKEN" to token
)
cookies = req.cookies
res.select("select#changeServer option").apmap { source ->
2023-09-23 22:05:33 +00:00
val server = source.attr("value")
2023-11-01 04:31:17 +00:00
val link = "$data?dfgRr1OagZvvxbzHNpyCy0FqJQ18mCnb=$st&twEvZlbZbYRWBdKKwxkOnwYF0VWoGGVg=$server"
2023-10-18 09:54:44 +00:00
if (server.contains(Regex("(?i)kuramadrive|archive"))) {
2023-09-23 22:05:33 +00:00
invokeLocalSource(link, server, data, callback)
} else {
app.get(
link,
referer = data,
headers = headers,
cookies = cookies
).document.select("div.iframe-container iframe").attr("src").let { videoUrl ->
loadExtractor(fixUrl(videoUrl), "$mainUrl/", subtitleCallback, callback)
}
2022-09-09 16:09:54 +00:00
}
}
return true
}
}