From 596ab11cd0fa8b262d36ea280c0629a5dc6d5010 Mon Sep 17 00:00:00 2001 From: CakesTwix Date: Mon, 5 Feb 2024 20:44:12 +0200 Subject: [PATCH] anitubeinua: Fix fucking csst -> monstro Check btw https://monstro.site/ and https://csst.online) @mail.ru moment on ukrainian site, lol --- AnitubeinuaProvider/build.gradle.kts | 2 +- .../com/lagradost/AnitubeinuaProvider.kt | 224 +++++++++--------- 2 files changed, 119 insertions(+), 107 deletions(-) diff --git a/AnitubeinuaProvider/build.gradle.kts b/AnitubeinuaProvider/build.gradle.kts index 8b5e69d..cf3008f 100644 --- a/AnitubeinuaProvider/build.gradle.kts +++ b/AnitubeinuaProvider/build.gradle.kts @@ -1,5 +1,5 @@ // use an integer for version numbers -version = 5 +version = 6 cloudstream { diff --git a/AnitubeinuaProvider/src/main/kotlin/com/lagradost/AnitubeinuaProvider.kt b/AnitubeinuaProvider/src/main/kotlin/com/lagradost/AnitubeinuaProvider.kt index f7cceb7..ac7baa0 100644 --- a/AnitubeinuaProvider/src/main/kotlin/com/lagradost/AnitubeinuaProvider.kt +++ b/AnitubeinuaProvider/src/main/kotlin/com/lagradost/AnitubeinuaProvider.kt @@ -1,20 +1,20 @@ package com.lagradost -import com.lagradost.models.PlayerJson -import com.lagradost.extractors.AshdiExtractor import com.lagradost.cloudstream3.* -import com.lagradost.cloudstream3.extractors.Mp4Upload import com.lagradost.cloudstream3.LoadResponse.Companion.addTrailer +import com.lagradost.cloudstream3.extractors.Mp4Upload import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson import com.lagradost.cloudstream3.utils.ExtractorLink import com.lagradost.cloudstream3.utils.M3u8Helper +import com.lagradost.extractors.AshdiExtractor import com.lagradost.extractors.csstExtractor import com.lagradost.models.Ajax import com.lagradost.models.Link +import com.lagradost.models.PlayerJson import com.lagradost.models.videoConstructor +import java.util.* import org.jsoup.Jsoup import org.jsoup.nodes.Element -import java.util.* class AnitubeinuaProvider : MainAPI() { @@ -24,37 +24,35 @@ class AnitubeinuaProvider : MainAPI() { override val hasMainPage = true override var lang = "uk" override val hasDownloadSupport = true - override val supportedTypes = setOf( - TvType.AnimeMovie, - TvType.Anime, - ) + override val supportedTypes = + setOf( + TvType.AnimeMovie, + TvType.Anime, + ) // Sections - override val mainPage = mainPageOf( - "$mainUrl/anime/page/" to "Нові", - "$mainUrl/f/sort=rating/order=desc/page/" to "Популярні", - ) + override val mainPage = + mainPageOf( + "$mainUrl/anime/page/" to "Нові", + "$mainUrl/f/sort=rating/order=desc/page/" to "Популярні", + ) - override suspend fun getMainPage( - page: Int, - request: MainPageRequest - ): HomePageResponse { + override suspend fun getMainPage(page: Int, request: MainPageRequest): HomePageResponse { val document = app.get(request.data + page).document - val home = document.select(".story").map { - it.toSearchResponse() - } + val home = document.select(".story").map { it.toSearchResponse() } return newHomePageResponse(request.name, home) } private fun Element.toSearchResponse(): AnimeSearchResponse { val title = this.selectFirst(".story_c h2 a, div.text_content a")?.text()?.trim().toString() val href = this.selectFirst(".story_c h2 a, div.text_content a")?.attr("href").toString() - val posterUrl = mainUrl + this.selectFirst(".story_c_l span.story_post img, a img")?.attr("src") + val posterUrl = + mainUrl + this.selectFirst(".story_c_l span.story_post img, a img")?.attr("src") var isSub = this.select(".box .sub").isNotEmpty() var isDub = this.select(".box .ukr").isNotEmpty() - if (!isSub && !isDub){ + if (!isSub && !isDub) { isSub = true isDub = true } @@ -62,22 +60,22 @@ class AnitubeinuaProvider : MainAPI() { this.posterUrl = posterUrl addDubStatus(isDub, isSub) } - } override suspend fun search(query: String): List { - val document = app.post( - url = mainUrl, - data = mapOf( - "do" to "search", - "subaction" to "search", - "story" to query.replace(" ", "+") + val document = + app.post( + url = mainUrl, + data = + mapOf( + "do" to "search", + "subaction" to "search", + "story" to query.replace(" ", "+") + ) ) - ).document + .document - return document.select("article.story").map { - it.toSearchResponse() - } + return document.select("article.story").map { it.toSearchResponse() } } // Detailed information @@ -88,7 +86,8 @@ class AnitubeinuaProvider : MainAPI() { // Parse info val title = document.selectFirst(".story_c h2")?.text()?.trim().toString() - val poster = mainUrl + document.selectFirst(".story_c_left span.story_post img")?.attr("src") + val poster = + mainUrl + document.selectFirst(".story_c_left span.story_post img")?.attr("src") val tags = someInfo.select("a[href*=/anime/]").map { it.text() } val year = someInfo.select("a[href*=/xfsearch/year/]").text().toIntOrNull() @@ -96,11 +95,10 @@ class AnitubeinuaProvider : MainAPI() { val description = document.selectFirst("div.my-text")?.text()?.trim() // val author = someInfo.select("strong:contains(Студія:)").next().html() val trailer = document.selectFirst(".rcol a.rollover")?.attr("href").toString() - val rating = document.selectFirst(".lexington-box > div:last-child span")?.text().toRatingInt() + val rating = + document.selectFirst(".lexington-box > div:last-child span")?.text().toRatingInt() - val recommendations = document.select(".horizontal ul li").map { - it.toSearchResponse() - } + val recommendations = document.select(".horizontal ul li").map { it.toSearchResponse() } // Return to app // Players, Episodes, Number of episodes @@ -108,42 +106,49 @@ class AnitubeinuaProvider : MainAPI() { val dubEpisodes = mutableListOf() val id = url.split("/").last().split("-").first() - val ajax = fromPlaylistAjax("$mainUrl/engine/ajax/playlists.php?news_id=$id&xfield=playlist&time=${Date().time}") + val ajax = + fromPlaylistAjax( + "$mainUrl/engine/ajax/playlists.php?news_id=$id&xfield=playlist&time=${Date().time}" + ) if (!ajax.isNullOrEmpty()) { // Ajax list - ajax.groupBy{ it.name }.forEach { episodes -> // Group by name - episodes.value.forEach lit@{ - // UFDub player, drop - if(it.name == "ПЛЕЙЛИСТ") return@lit + ajax + .groupBy { it.name } + .forEach { episodes -> // Group by name + episodes.value.forEach lit@{ + // UFDub player, drop + if (it.name == "ПЛЕЙЛИСТ") return@lit - if(it.urls.isDub){ - dubEpisodes.add(Episode( - "${it.name}, $id, ${it.urls.isDub}", - it.name, - episode = it.numberEpisode + if (it.urls.isDub) { + dubEpisodes.add( + Episode( + "${it.name}, $id, ${it.urls.isDub}", + it.name, + episode = it.numberEpisode + ) ) - ) - } else { - subEpisodes.add(Episode( - "${it.name}, $id, ${it.urls.isDub}", - it.name, - episode = it.numberEpisode + } else { + subEpisodes.add( + Episode( + "${it.name}, $id, ${it.urls.isDub}", + it.name, + episode = it.numberEpisode + ) ) - ) + } } } - } } else { - document.select("script").map{ script -> + document.select("script").map { script -> if (script.data().contains("RalodePlayer.init(")) { val episodesList = fromVideoContructor(script) episodesList.forEach { episode -> // UFDub player, drop var varEpisodeNumber = episode.episodeNumber - if(episode.episodeName == "ПЛЕЙЛИСТ") return@forEach - if(varEpisodeNumber == null){ - varEpisodeNumber = episodesList.last().episodeNumber?.plus(1); + if (episode.episodeName == "ПЛЕЙЛИСТ") return@forEach + if (varEpisodeNumber == null) { + varEpisodeNumber = episodesList.last().episodeNumber?.plus(1) } dubEpisodes.add( Episode( @@ -152,10 +157,8 @@ class AnitubeinuaProvider : MainAPI() { episode = varEpisodeNumber, ) ) - } } - } } @@ -172,7 +175,6 @@ class AnitubeinuaProvider : MainAPI() { } } - // It works when I click to view the series override suspend fun loadLinks( data: String, // (Ajax) Name, id title, isDub | (Two) Episode name, url title @@ -181,30 +183,37 @@ class AnitubeinuaProvider : MainAPI() { callback: (ExtractorLink) -> Unit ): Boolean { val dataList = data.split(", ") - if(dataList[1].toIntOrNull() != null){ // Its ajax list + // Log.d("CakesTwix-Debug", data) + if (dataList[1].toIntOrNull() != null) { // Its ajax list - val ajax = fromPlaylistAjax("$mainUrl/engine/ajax/playlists.php?news_id=${dataList[1]}&xfield=playlist&time=${Date().time}") + val ajax = + fromPlaylistAjax( + "$mainUrl/engine/ajax/playlists.php?news_id=${dataList[1]}&xfield=playlist&time=${Date().time}" + ) // Filter by name and isDub - ajax?.filter { it.name == dataList[0] } + ajax + ?.filter { it.name == dataList[0] } ?.filter { it.urls.isDub == dataList[2].toBoolean() } ?.forEach { // Get m3u8 url - with(it){ - when{ + with(it) { + when { it.urls.url.contains("https://tortuga.wtf/vod/") -> { M3u8Helper.generateM3u8( source = "${it.urls.playerName} (${it.urls.name})", streamUrl = AshdiExtractor().ParseM3U8(this.urls.url), referer = "https://tortuga.wtf/" - ).forEach(callback) + ) + .forEach(callback) } it.urls.url.contains("https://ashdi.vip/vod") -> { M3u8Helper.generateM3u8( source = "${it.urls.playerName} (${it.urls.name})", streamUrl = AshdiExtractor().ParseM3U8(this.urls.url), referer = "https://qeruya.cyou" - ).forEach(callback) + ) + .forEach(callback) } it.urls.url.contains("https://www.udrop.com") -> { callback.invoke( @@ -218,8 +227,9 @@ class AnitubeinuaProvider : MainAPI() { ) ) } - it.urls.url.contains("https://csst.online/embed/") -> { - csstExtractor().ParseUrl(it.urls.url).split(",").forEach{ csstUrl -> + it.urls.url.contains("https://csst.online/embed/") || it.urls.url.contains("https://monstro.site/embed/") -> { + csstExtractor().ParseUrl(it.urls.url).split(",").forEach { csstUrl + -> callback.invoke( ExtractorLink( this.urls.url, @@ -233,7 +243,7 @@ class AnitubeinuaProvider : MainAPI() { } } it.urls.url.contains("https://www.mp4upload.com/") -> { - Mp4Upload().getUrl(it.urls.url)?.forEach{ extlink -> + Mp4Upload().getUrl(it.urls.url)?.forEach { extlink -> callback.invoke( ExtractorLink( extlink.source, @@ -243,7 +253,6 @@ class AnitubeinuaProvider : MainAPI() { extlink.quality, extlink.type, extlink.headers, - ) ) } @@ -252,22 +261,22 @@ class AnitubeinuaProvider : MainAPI() { } } } - } else { + } else { val document = app.get(dataList[1]).document document.select("script").map { script -> if (script.data().contains("RalodePlayer.init(")) { var latestNumber: Int? = 0 fromVideoContructor(script).forEach { dub -> - if(dub.episodeName == "ПЛЕЙЛИСТ") return@forEach + if (dub.episodeName == "ПЛЕЙЛИСТ") return@forEach // Parse by number episode // If null, just add +1 - if(dub.episodeNumber == null){ - dub.episodeNumber = latestNumber?.plus(1); + if (dub.episodeNumber == null) { + dub.episodeNumber = latestNumber?.plus(1) } latestNumber = dub.episodeNumber - if(latestNumber != dataList[0].toIntOrNull()) return@forEach + if (latestNumber != dataList[0].toIntOrNull()) return@forEach with(dub.episodeUrl) { when { @@ -276,14 +285,16 @@ class AnitubeinuaProvider : MainAPI() { source = dub.playerName, streamUrl = AshdiExtractor().ParseM3U8(this), referer = "https://tortuga.wtf/" - ).forEach(callback) + ) + .forEach(callback) } contains("https://ashdi.vip/vod") -> { M3u8Helper.generateM3u8( source = dub.playerName, streamUrl = AshdiExtractor().ParseM3U8(this), referer = "https://qeruya.cyou" - ).forEach(callback) + ) + .forEach(callback) } contains("https://www.udrop.com") -> { callback.invoke( @@ -297,12 +308,12 @@ class AnitubeinuaProvider : MainAPI() { ) ) } - contains("https://csst.online/embed/") -> { + contains("https://monstro.site/embed/") || contains("https://csst.online/embed/") -> { csstExtractor().ParseUrl(this).split(",").forEach { callback.invoke( ExtractorLink( dub.playerName, - name = "${dub.playerName} ${it.substringBefore("]").drop(1)}", + name = "${dub.playerName.replace("\"", "")} ${it.substringBefore("]").drop(1)}", it.substringAfter("]"), "", 0, @@ -312,7 +323,7 @@ class AnitubeinuaProvider : MainAPI() { } } contains("https://www.mp4upload.com/") -> { - Mp4Upload().getUrl(this)?.forEach{ extlink -> + Mp4Upload().getUrl(this)?.forEach { extlink -> callback.invoke( ExtractorLink( extlink.source, @@ -322,8 +333,7 @@ class AnitubeinuaProvider : MainAPI() { extlink.quality, extlink.type, extlink.headers, - - ) + ) ) } } @@ -337,7 +347,7 @@ class AnitubeinuaProvider : MainAPI() { return true } - private fun decode(input: String): String{ + private fun decode(input: String): String { // Decoded string, thanks to Secozzi val hexRegex = Regex("\\\\u([0-9a-fA-F]{4})") return hexRegex.replace(input) { matchResult -> @@ -345,11 +355,7 @@ class AnitubeinuaProvider : MainAPI() { } } - data class Responses( - val success: Boolean?, - val response: String?, - val message: String? - ) + data class Responses(val success: Boolean?, val response: String?, val message: String?) // Thanks to Andro999b // https://github.com/Andro999b/movies-telegram-bot/blob/a296c7d4122a25fa70b612e75d741dd55c154640/functions/src/providers/AnitubeUAProvider.ts#L86-L137 @@ -357,7 +363,7 @@ class AnitubeinuaProvider : MainAPI() { val responseGet = app.get(url).parsedSafe() // Not Ajax, return null - if(responseGet?.success == false){ + if (responseGet?.success == false) { // Log.d("load-debug", "Not Ajax") return null } @@ -365,15 +371,15 @@ class AnitubeinuaProvider : MainAPI() { val returnEpisodes = mutableListOf() val playlist = Jsoup.parse(responseGet?.response!!) - val audios = mutableListOf>() // (INARI, 0_0, ...) - val listDubStatus = mutableListOf>() // (СУБТИТРИ, 0_0_0, ...) - val listPlayers = mutableListOf>() // (ПЛЕЄР МОНСТР, 0_0_0_0, ...) + val audios = mutableListOf>() // (INARI, 0_0, ...) + val listDubStatus = mutableListOf>() // (СУБТИТРИ, 0_0_0, ...) + val listPlayers = mutableListOf>() // (ПЛЕЄР МОНСТР, 0_0_0_0, ...) playlist.select(".playlists-lists .playlists-items:first-child li").forEach { audios.add(Pair(it.text(), it.attr("data-id"))) } // Set listPlayers and listDubStatus // If has subs - So players in 3 index - if(playlist.select(".playlists-lists .playlists-items").count() == 3){ + if (playlist.select(".playlists-lists .playlists-items").count() == 3) { // Players playlist.select(".playlists-lists .playlists-items:nth-child(3) li").forEach { listPlayers.add(Pair(it.text(), it.attr("data-id"))) @@ -401,15 +407,15 @@ class AnitubeinuaProvider : MainAPI() { // Set this element Dub name audios.forEach { - if(audioId.startsWith(it.second)){ + if (audioId.startsWith(it.second)) { audio = it.first } } - if(audioId.count { it == '_' } == 3){ + if (audioId.count { it == '_' } == 3) { listDubStatus.forEach { - if(audioId.startsWith(it.second)){ - if(it.first == "СУБТИТРИ"){ + if (audioId.startsWith(it.second)) { + if (it.first == "СУБТИТРИ") { isDub = false } } @@ -417,7 +423,7 @@ class AnitubeinuaProvider : MainAPI() { } listPlayers.forEach { - if(audioId.startsWith(it.second)){ + if (audioId.startsWith(it.second)) { playerName = it.first } } @@ -441,8 +447,14 @@ class AnitubeinuaProvider : MainAPI() { private fun fromVideoContructor(script: Element): List { val playerScriptRawJson = script.data().substringAfterLast(".init(").substringBefore(");") - val playerEpisodesRawJson = playerScriptRawJson.substringAfter("],").substringBeforeLast(",") - val playerNamesArray = (playerScriptRawJson.substringBefore("],") + "]").dropLast(1).drop(1).replace("\",\"", ",,,").split(",,,") + val playerEpisodesRawJson = + playerScriptRawJson.substringAfter("],").substringBeforeLast(",") + val playerNamesArray = + (playerScriptRawJson.substringBefore("],") + "]") + .dropLast(1) + .drop(1) + .replace("\",\"", ",,,") + .split(",,,") // val numberOfEpisodesInt = playerScriptRawJson.substringAfterLast(",").toIntOrNull() val jsonEpisodes = tryParseJson>>(playerEpisodesRawJson)!! @@ -450,7 +462,7 @@ class AnitubeinuaProvider : MainAPI() { jsonEpisodes.forEachIndexed { index, episode -> val playerName = decode(playerNamesArray[index]) - episode.forEach{ + episode.forEach { episodes.add( videoConstructor( playerName, @@ -464,13 +476,13 @@ class AnitubeinuaProvider : MainAPI() { } return episodes.toList() } + private fun extractIntFromString(string: String): Int? { val value = Regex("(\\d+)").findAll(string).lastOrNull() ?: return null - if(value.value[0].toString() == "0"){ + if (value.value[0].toString() == "0") { return value.value.drop(1).toIntOrNull() } return value.value.toIntOrNull() - } }