anitubeinua: refactor ajax parser

This commit is contained in:
CakesTwix 2023-03-11 17:59:23 +02:00
parent 599fda2d0b
commit 96f26e933c
Signed by: CakesTwix
GPG key ID: 7B11051D5CE19825
3 changed files with 190 additions and 107 deletions

View file

@ -1,5 +1,5 @@
// use an integer for version numbers // use an integer for version numbers
version = 2 version = 3
cloudstream { cloudstream {

View file

@ -1,5 +1,6 @@
package com.lagradost package com.lagradost
import android.util.Log
import com.lagradost.models.PlayerJson import com.lagradost.models.PlayerJson
import com.lagradost.extractors.AshdiExtractor import com.lagradost.extractors.AshdiExtractor
import com.lagradost.cloudstream3.* import com.lagradost.cloudstream3.*
@ -7,6 +8,8 @@ import com.lagradost.cloudstream3.utils.AppUtils.tryParseJson
import com.lagradost.cloudstream3.utils.ExtractorLink import com.lagradost.cloudstream3.utils.ExtractorLink
import com.lagradost.cloudstream3.utils.M3u8Helper import com.lagradost.cloudstream3.utils.M3u8Helper
import com.lagradost.extractors.csstExtractor import com.lagradost.extractors.csstExtractor
import com.lagradost.models.Ajax
import com.lagradost.models.Link
import org.jsoup.Jsoup import org.jsoup.Jsoup
import org.jsoup.nodes.Element import org.jsoup.nodes.Element
import java.util.* import java.util.*
@ -76,7 +79,7 @@ class AnitubeinuaProvider : MainAPI() {
} }
// Detailed information // Detailed information
override suspend fun load(url: String): LoadResponse { override suspend fun load(url: String): AnimeLoadResponse {
val document = app.get(url).document val document = app.get(url).document
val someInfo = document.select("div.story_c_r")[1] val someInfo = document.select("div.story_c_r")[1]
@ -98,25 +101,27 @@ class AnitubeinuaProvider : MainAPI() {
// Return to app // Return to app
// Players, Episodes, Number of episodes // Players, Episodes, Number of episodes
var episodes: List<Episode> = emptyList() var subEpisodes = mutableListOf<Episode>()
var dubEpisodes = mutableListOf<Episode>()
val id = url.split("/").last().split("-").first() val id = url.split("/").last().split("-").first()
val responseGet = app.get("$mainUrl/engine/ajax/playlists.php?news_id=$id&xfield=playlist&time=${Date().time}").parsedSafe<Responses>()!!
if (responseGet.success == true) { // First type players val ajax = fromPlaylistAjax("$mainUrl/engine/ajax/playlists.php?news_id=$id&xfield=playlist&time=${Date().time}")
episodes =
app.get("$mainUrl/engine/ajax/playlists.php?news_id=$id&xfield=playlist&time=${Date().time}") if (!ajax.isNullOrEmpty()) { // Ajax list
.parsedSafe<Responses>()?.response.let { ajax.groupBy{ it.name }.forEach { episodes -> // Group by name
Jsoup.parse(it.toString()).select("div.playlists-videos li") episodes.value.forEach{
.mapIndexedNotNull() { index, eps -> if(it.urls.isDub){
val href = dubEpisodes.add(Episode(
"$mainUrl/engine/ajax/playlists.php?news_id=$id&xfield=playlist&time=${Date().time}" "${it.name}, $id, ${it.urls.isDub}",
val name = eps.text().trim() // Серія 1 it.name,
if (href.isNotEmpty()) { episode = it.numberEpisode)
Episode(
"$href, $index", // link, Серія 1
name,
) )
} else { } else {
null subEpisodes.add(Episode(
"${it.name}, $id, ${it.urls.isDub}",
it.name,
episode = it.numberEpisode)
)
} }
} }
} }
@ -133,7 +138,7 @@ class AnitubeinuaProvider : MainAPI() {
item.forEachIndexed { index, item2 -> item.forEachIndexed { index, item2 ->
if(!item2.name.contains("ПЛЕЙЛИСТ")) // UFDub player if(!item2.name.contains("ПЛЕЙЛИСТ")) // UFDub player
{ {
episodes = episodes.plus( dubEpisodes.add(
Episode( Episode(
"$index, $url", "$index, $url",
item2.name, item2.name,
@ -148,101 +153,72 @@ class AnitubeinuaProvider : MainAPI() {
} }
} }
return newTvSeriesLoadResponse(title, url, tvType, episodes.distinctBy{ it.name }) { return newAnimeLoadResponse(title, url, tvType) {
this.posterUrl = poster this.posterUrl = poster
this.year = year this.year = year
this.plot = description this.plot = description
this.tags = tags this.tags = tags
this.rating = rating this.rating = rating
this.recommendations = recommendations this.recommendations = recommendations
addEpisodes(DubStatus.Dubbed, dubEpisodes)
addEpisodes(DubStatus.Subbed, subEpisodes)
} }
} }
// It works when I click to view the series // It works when I click to view the series
override suspend fun loadLinks( override suspend fun loadLinks(
data: String, // (First) link, index | (Two) index, url title data: String, // (Ajax) Name, id title, isDub | (Two) index, url title
isCasting: Boolean, isCasting: Boolean,
subtitleCallback: (SubtitleFile) -> Unit, subtitleCallback: (SubtitleFile) -> Unit,
callback: (ExtractorLink) -> Unit callback: (ExtractorLink) -> Unit
): Boolean { ): Boolean {
val dataList = data.split(", ") val dataList = data.split(", ")
if(dataList[0].contains("https://")){ // Its First type player if(dataList[0].toIntOrNull() == null){ // Its ajax list
// Get episodes list (as json) val ajax = fromPlaylistAjax("$mainUrl/engine/ajax/playlists.php?news_id=${dataList[1]}&xfield=playlist&time=${Date().time}")
val responseGet = app.get(dataList[0]).parsedSafe<Responses>() // ajax link
responseGet?.response?.let { it ->
// List Players // Filter by name and isDub
val playersTab = Jsoup.parse(it).select("div.playlists-items") ajax?.filter { it.name == dataList[0] }
?.filter { it.urls.isDub == dataList[2].toBoolean() }
?.forEach {
// Parse all episodes by name // Get m3u8 url
var index = 0 with(it){
var playerTabId = ""
Jsoup.parse(it).select("div.playlists-videos li")
.mapNotNull { eps ->
// 0 - idk, 1 - dub, 2 - player
// if with sub
// 0 - idk 1 - dub 2 - sub or dub 3 - player
// dataList[1] - index
// 0_1_2
// Log.d("load-debug", eps.attr("data-id"))
if(playerTabId != eps.attr("data-id")){
index = 0
playerTabId = eps.attr("data-id")
}
if(dataList[1].toInt() == index){
var href = eps.attr("data-file") // m3u url
// Can be without https:
if (!href.contains("https://")) {
href = "https:$href"
}
val dubName = playersTab[0].select(" li[data-id=${ playerTabId.take(3) }]").text() // G&M
var playerName = playersTab[1].select(" li[data-id=$playerTabId]").text() // ПЛЕЄР ASHDI
if(playerTabId.count { it == '_' } == 3) {
playerName =
playersTab[2].selectFirst(" li[data-id=$playerTabId]")!!
.text() // ПЛЕЄР ASHDI
}
with(href){
when{ when{
contains("https://tortuga.wtf/vod/") -> { it.urls.url.contains("https://tortuga.wtf/vod/") -> {
M3u8Helper.generateM3u8( M3u8Helper.generateM3u8(
source = "$playerName ($dubName)", source = "${it.urls.playerName} (${it.urls.name})",
streamUrl = AshdiExtractor().ParseM3U8(href), streamUrl = AshdiExtractor().ParseM3U8(this.urls.url),
referer = "https://tortuga.wtf/" referer = "https://tortuga.wtf/"
).forEach(callback) ).forEach(callback)
} }
contains("https://ashdi.vip/vod") -> { it.urls.url.contains("https://ashdi.vip/vod") -> {
M3u8Helper.generateM3u8( M3u8Helper.generateM3u8(
source = "$playerName ($dubName)", source = "${it.urls.playerName} (${it.urls.name})",
streamUrl = AshdiExtractor().ParseM3U8(href), streamUrl = AshdiExtractor().ParseM3U8(this.urls.url),
referer = "https://qeruya.cyou" referer = "https://qeruya.cyou"
).forEach(callback) ).forEach(callback)
} }
contains("https://www.udrop.com") -> { it.urls.url.contains("https://www.udrop.com") -> {
callback.invoke( callback.invoke(
ExtractorLink( ExtractorLink(
"$playerName ($dubName)", this.urls.url,
name = "$playerName ($dubName)", name = "${it.urls.playerName} (${it.urls.name})",
href, this.urls.url,
"", "",
0, 0,
isM3u8 = false, isM3u8 = false,
) )
) )
} }
contains("https://csst.online/embed/") -> { it.urls.url.contains("https://csst.online/embed/") -> {
csstExtractor().ParseUrl(href).split(",").forEach{ csstExtractor().ParseUrl(it.urls.url).split(",").forEach{ csstUrl ->
callback.invoke( callback.invoke(
ExtractorLink( ExtractorLink(
"$playerName ($dubName)", this.urls.url,
name = "$playerName ($dubName) ${it.substringBefore("]").drop(1)}", name = "${it.urls.playerName} (${it.urls.name}) ${csstUrl.substringBefore("]").drop(1)}",
it.substringAfter("]"), csstUrl.substringAfter("]"),
"", "",
0, 0,
isM3u8 = false, isM3u8 = false,
@ -254,9 +230,6 @@ class AnitubeinuaProvider : MainAPI() {
} }
} }
} }
index += 1
}
}
} else { } else {
val document = app.get(dataList[1]).document val document = app.get(dataList[1]).document
document.select("script").map { script -> document.select("script").map { script ->
@ -336,4 +309,99 @@ class AnitubeinuaProvider : MainAPI() {
val response: String?, val response: String?,
val message: String? val message: String?
) )
private suspend fun fromPlaylistAjax(url: String): List<Ajax>? {
val responseGet = app.get(url).parsedSafe<Responses>()
// Not Ajax, return null
if(responseGet?.success == false){
// Log.d("load-debug", "Not Ajax")
return null
}
val returnEpisodes = mutableListOf<Ajax>()
val playlist = Jsoup.parse(responseGet?.response!!)
val audios = mutableListOf<Pair<String,String>>() // (INARI, 0_0, ...)
val listDubStatus = mutableListOf<Pair<String,String>>() // (СУБТИТРИ, 0_0_0, ...)
val listPlayers = mutableListOf<Pair<String,String>>() // (ПЛЕЄР МОНСТР, 0_0_0_0, ...)
playlist.select(".playlists-lists .playlists-items:first-child li").forEach {
audios.add(Pair(it.text(), it.attr("data-id")))
}
// If has subs - So players in 3 index
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")))
}
// Subs/Dubs index
playlist.select(".playlists-lists .playlists-items:nth-child(2) li").forEach {
listDubStatus.add(Pair(it.text(), it.attr("data-id")))
}
} else {
// Players
playlist.select(".playlists-lists .playlists-items:nth-child(2) li").forEach {
listPlayers.add(Pair(it.text(), it.attr("data-id")))
}
}
playlist.select(".playlists-videos .playlists-items li").forEach { element ->
val audioId = element.attr("data-id") // 0_0_0 or 0_0_0_0 if subs
val episodeId = extractIntFromString(element.text())
val url = element.attr("data-file")
var isDub = true
var audio: String? = null
var playerName = ""
// Set this element Dub name
audios.forEach {
if(audioId.startsWith(it.second)){
audio = it.first
}
}
if(audioId.count { it == '_' } == 3){
listDubStatus.forEach {
if(audioId.startsWith(it.second)){
if(it.first == "СУБТИТРИ"){
isDub = false
}
}
}
listPlayers.forEach {
if(audioId.startsWith(it.second)){
playerName = it.first
}
}
}
returnEpisodes.add(
Ajax(
episodeId,
element.text(),
Link(
isDub, // TODO: Impl
url,
audio.toString(),
playerName,
)
))
}
return returnEpisodes.toList()
}
private fun extractIntFromString(string: String): Int? {
val value = Regex("(\\d+)").find(string)?.value
if(value!![0].toString() == "0"){
return value.drop(1).toIntOrNull()
}
return value.toIntOrNull()
}
} }

View file

@ -0,0 +1,15 @@
package com.lagradost.models
data class Ajax (
val numberEpisode : Int?,
val name : String,
val urls : Link,
)
data class Link (
val isDub : Boolean,
val url : String,
val name : String,
val playerName : String,
)